HBASE-3287 Add option to cache blocks on hfile write and evict blocks on hfile close
git-svn-id: https://svn.apache.org/repos/asf/hbase/trunk@1040762 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
70f5422d72
commit
826fbd99ff
|
@ -16,6 +16,10 @@ Release 0.91.0 - Unreleased
|
||||||
HBASE-2001 Coprocessors: Colocate user code with regions (Mingjie Lai via
|
HBASE-2001 Coprocessors: Colocate user code with regions (Mingjie Lai via
|
||||||
Andrew Purtell)
|
Andrew Purtell)
|
||||||
|
|
||||||
|
NEW FEATURES
|
||||||
|
HBASE-3287 Add option to cache blocks on hfile write and evict blocks on
|
||||||
|
hfile close
|
||||||
|
|
||||||
|
|
||||||
Release 0.90.0 - Unreleased
|
Release 0.90.0 - Unreleased
|
||||||
INCOMPATIBLE CHANGES
|
INCOMPATIBLE CHANGES
|
||||||
|
|
|
@ -62,7 +62,7 @@ public class HalfStoreFileReader extends StoreFile.Reader {
|
||||||
public HalfStoreFileReader(final FileSystem fs, final Path p, final BlockCache c,
|
public HalfStoreFileReader(final FileSystem fs, final Path p, final BlockCache c,
|
||||||
final Reference r)
|
final Reference r)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
super(fs, p, c, false);
|
super(fs, p, c, false, false);
|
||||||
// This is not actual midkey for this half-file; its just border
|
// This is not actual midkey for this half-file; its just border
|
||||||
// around which we split top and bottom. Have to look in files to find
|
// around which we split top and bottom. Have to look in files to find
|
||||||
// actual last and first keys for bottom and top halves. Half-files don't
|
// actual last and first keys for bottom and top halves. Half-files don't
|
||||||
|
|
|
@ -21,6 +21,8 @@ package org.apache.hadoop.hbase.io.hfile;
|
||||||
|
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
|
|
||||||
|
import org.apache.hadoop.hbase.io.hfile.LruBlockCache.CacheStats;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Block cache interface.
|
* Block cache interface.
|
||||||
* TODO: Add filename or hash of filename to block cache key.
|
* TODO: Add filename or hash of filename to block cache key.
|
||||||
|
@ -49,6 +51,19 @@ public interface BlockCache {
|
||||||
*/
|
*/
|
||||||
public ByteBuffer getBlock(String blockName, boolean caching);
|
public ByteBuffer getBlock(String blockName, boolean caching);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Evict block from cache.
|
||||||
|
* @param blockName Block name to evict
|
||||||
|
* @return true if block existed and was evicted, false if not
|
||||||
|
*/
|
||||||
|
public boolean evictBlock(String blockName);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the statistics for this block cache.
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public CacheStats getStats();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Shutdown the cache.
|
* Shutdown the cache.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -20,10 +20,11 @@
|
||||||
package org.apache.hadoop.hbase.io.hfile;
|
package org.apache.hadoop.hbase.io.hfile;
|
||||||
|
|
||||||
import java.io.BufferedInputStream;
|
import java.io.BufferedInputStream;
|
||||||
|
import java.io.ByteArrayInputStream;
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
import java.io.Closeable;
|
import java.io.Closeable;
|
||||||
import java.io.DataInputStream;
|
import java.io.DataInputStream;
|
||||||
import java.io.DataOutputStream;
|
import java.io.DataOutputStream;
|
||||||
import java.io.ByteArrayInputStream;
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
|
@ -260,6 +261,14 @@ public class HFile {
|
||||||
// May be null if we were passed a stream.
|
// May be null if we were passed a stream.
|
||||||
private Path path = null;
|
private Path path = null;
|
||||||
|
|
||||||
|
// Block cache to optionally fill on write
|
||||||
|
private BlockCache blockCache;
|
||||||
|
|
||||||
|
// Additional byte array output stream used to fill block cache
|
||||||
|
private ByteArrayOutputStream baos;
|
||||||
|
private DataOutputStream baosDos;
|
||||||
|
private int blockNumber = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor that uses all defaults for compression and block size.
|
* Constructor that uses all defaults for compression and block size.
|
||||||
* @param fs
|
* @param fs
|
||||||
|
@ -268,7 +277,8 @@ public class HFile {
|
||||||
*/
|
*/
|
||||||
public Writer(FileSystem fs, Path path)
|
public Writer(FileSystem fs, Path path)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
this(fs, path, DEFAULT_BLOCKSIZE, (Compression.Algorithm) null, null);
|
this(fs, path, DEFAULT_BLOCKSIZE, (Compression.Algorithm) null, null,
|
||||||
|
null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -287,7 +297,7 @@ public class HFile {
|
||||||
this(fs, path, blocksize,
|
this(fs, path, blocksize,
|
||||||
compress == null? DEFAULT_COMPRESSION_ALGORITHM:
|
compress == null? DEFAULT_COMPRESSION_ALGORITHM:
|
||||||
Compression.getCompressionAlgorithmByName(compress),
|
Compression.getCompressionAlgorithmByName(compress),
|
||||||
comparator);
|
comparator, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -301,12 +311,13 @@ public class HFile {
|
||||||
*/
|
*/
|
||||||
public Writer(FileSystem fs, Path path, int blocksize,
|
public Writer(FileSystem fs, Path path, int blocksize,
|
||||||
Compression.Algorithm compress,
|
Compression.Algorithm compress,
|
||||||
final KeyComparator comparator)
|
final KeyComparator comparator, BlockCache blockCache)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
this(fs.create(path), blocksize, compress, comparator);
|
this(fs.create(path), blocksize, compress, comparator);
|
||||||
this.closeOutputStream = true;
|
this.closeOutputStream = true;
|
||||||
this.name = path.toString();
|
this.name = path.toString();
|
||||||
this.path = path;
|
this.path = path;
|
||||||
|
this.blockCache = blockCache;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -371,6 +382,17 @@ public class HFile {
|
||||||
|
|
||||||
writeTime += System.currentTimeMillis() - now;
|
writeTime += System.currentTimeMillis() - now;
|
||||||
writeOps++;
|
writeOps++;
|
||||||
|
|
||||||
|
if (blockCache != null) {
|
||||||
|
baosDos.flush();
|
||||||
|
byte [] bytes = baos.toByteArray();
|
||||||
|
ByteBuffer blockToCache = ByteBuffer.wrap(bytes, DATABLOCKMAGIC.length,
|
||||||
|
bytes.length - DATABLOCKMAGIC.length);
|
||||||
|
String blockName = path.toString() + blockNumber;
|
||||||
|
blockCache.cacheBlock(blockName, blockToCache);
|
||||||
|
baosDos.close();
|
||||||
|
}
|
||||||
|
blockNumber++;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -383,6 +405,11 @@ public class HFile {
|
||||||
this.out = getCompressingStream();
|
this.out = getCompressingStream();
|
||||||
this.out.write(DATABLOCKMAGIC);
|
this.out.write(DATABLOCKMAGIC);
|
||||||
firstKey = null;
|
firstKey = null;
|
||||||
|
if (blockCache != null) {
|
||||||
|
this.baos = new ByteArrayOutputStream();
|
||||||
|
this.baosDos = new DataOutputStream(baos);
|
||||||
|
this.baosDos.write(DATABLOCKMAGIC);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -552,6 +579,13 @@ public class HFile {
|
||||||
this.lastKeyOffset = koffset;
|
this.lastKeyOffset = koffset;
|
||||||
this.lastKeyLength = klength;
|
this.lastKeyLength = klength;
|
||||||
this.entryCount ++;
|
this.entryCount ++;
|
||||||
|
// If we are pre-caching blocks on write, fill byte array stream
|
||||||
|
if (blockCache != null) {
|
||||||
|
this.baosDos.writeInt(klength);
|
||||||
|
this.baosDos.writeInt(vlength);
|
||||||
|
this.baosDos.write(key, koffset, klength);
|
||||||
|
this.baosDos.write(value, voffset, vlength);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -729,6 +763,9 @@ public class HFile {
|
||||||
// Whether file is from in-memory store
|
// Whether file is from in-memory store
|
||||||
private boolean inMemory = false;
|
private boolean inMemory = false;
|
||||||
|
|
||||||
|
// Whether blocks of file should be evicted on close of file
|
||||||
|
private final boolean evictOnClose;
|
||||||
|
|
||||||
// Name for this object used when logging or in toString. Is either
|
// Name for this object used when logging or in toString. Is either
|
||||||
// the result of a toString on the stream or else is toString of passed
|
// the result of a toString on the stream or else is toString of passed
|
||||||
// file Path plus metadata key/value pairs.
|
// file Path plus metadata key/value pairs.
|
||||||
|
@ -743,9 +780,11 @@ public class HFile {
|
||||||
* @param cache block cache. Pass null if none.
|
* @param cache block cache. Pass null if none.
|
||||||
* @throws IOException
|
* @throws IOException
|
||||||
*/
|
*/
|
||||||
public Reader(FileSystem fs, Path path, BlockCache cache, boolean inMemory)
|
public Reader(FileSystem fs, Path path, BlockCache cache, boolean inMemory,
|
||||||
|
boolean evictOnClose)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
this(fs.open(path), fs.getFileStatus(path).getLen(), cache, inMemory);
|
this(path, fs.open(path), fs.getFileStatus(path).getLen(), cache,
|
||||||
|
inMemory, evictOnClose);
|
||||||
this.closeIStream = true;
|
this.closeIStream = true;
|
||||||
this.name = path.toString();
|
this.name = path.toString();
|
||||||
}
|
}
|
||||||
|
@ -758,16 +797,20 @@ public class HFile {
|
||||||
* stream.
|
* stream.
|
||||||
* @param size Length of the stream.
|
* @param size Length of the stream.
|
||||||
* @param cache block cache. Pass null if none.
|
* @param cache block cache. Pass null if none.
|
||||||
|
* @param inMemory whether blocks should be marked as in-memory in cache
|
||||||
|
* @param evictOnClose whether blocks in cache should be evicted on close
|
||||||
* @throws IOException
|
* @throws IOException
|
||||||
*/
|
*/
|
||||||
public Reader(final FSDataInputStream fsdis, final long size,
|
public Reader(Path path, final FSDataInputStream fsdis, final long size,
|
||||||
final BlockCache cache, final boolean inMemory) {
|
final BlockCache cache, final boolean inMemory,
|
||||||
|
final boolean evictOnClose) {
|
||||||
this.cache = cache;
|
this.cache = cache;
|
||||||
this.fileSize = size;
|
this.fileSize = size;
|
||||||
this.istream = fsdis;
|
this.istream = fsdis;
|
||||||
this.closeIStream = false;
|
this.closeIStream = false;
|
||||||
this.name = this.istream == null? "": this.istream.toString();
|
this.name = path.toString();
|
||||||
this.inMemory = inMemory;
|
this.inMemory = inMemory;
|
||||||
|
this.evictOnClose = evictOnClose;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -1192,6 +1235,14 @@ public class HFile {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void close() throws IOException {
|
public void close() throws IOException {
|
||||||
|
if (evictOnClose && this.cache != null) {
|
||||||
|
int numEvicted = 0;
|
||||||
|
for (int i=0; i<blockIndex.count; i++) {
|
||||||
|
if (this.cache.evictBlock(name + i)) numEvicted++;
|
||||||
|
}
|
||||||
|
LOG.debug("On close of file " + name + " evicted " + numEvicted +
|
||||||
|
" block(s) of " + blockIndex.count + " total blocks");
|
||||||
|
}
|
||||||
if (this.closeIStream && this.istream != null) {
|
if (this.closeIStream && this.istream != null) {
|
||||||
this.istream.close();
|
this.istream.close();
|
||||||
this.istream = null;
|
this.istream = null;
|
||||||
|
@ -1898,7 +1949,7 @@ public class HFile {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
// create reader and load file info
|
// create reader and load file info
|
||||||
HFile.Reader reader = new HFile.Reader(fs, file, null, false);
|
HFile.Reader reader = new HFile.Reader(fs, file, null, false, false);
|
||||||
Map<byte[],byte[]> fileInfo = reader.loadFileInfo();
|
Map<byte[],byte[]> fileInfo = reader.loadFileInfo();
|
||||||
// scan over file and read key/value's and check if requested
|
// scan over file and read key/value's and check if requested
|
||||||
HFileScanner scanner = reader.getScanner(false, false);
|
HFileScanner scanner = reader.getScanner(false, false);
|
||||||
|
|
|
@ -290,6 +290,15 @@ public class LruBlockCache implements BlockCache, HeapSize {
|
||||||
return cb.getBuffer();
|
return cb.getBuffer();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean evictBlock(String blockName) {
|
||||||
|
CachedBlock cb = map.get(blockName);
|
||||||
|
if (cb == null) return false;
|
||||||
|
evictBlock(cb);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
protected long evictBlock(CachedBlock block) {
|
protected long evictBlock(CachedBlock block) {
|
||||||
map.remove(block.getName());
|
map.remove(block.getName());
|
||||||
size.addAndGet(-1 * block.heapSize());
|
size.addAndGet(-1 * block.heapSize());
|
||||||
|
|
|
@ -25,6 +25,8 @@ import java.nio.ByteBuffer;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.apache.hadoop.hbase.io.hfile.LruBlockCache.CacheStats;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Simple one RFile soft reference cache.
|
* Simple one RFile soft reference cache.
|
||||||
|
@ -83,7 +85,18 @@ public class SimpleBlockCache implements BlockCache {
|
||||||
cache.put(blockName, new Ref(blockName, buf, q));
|
cache.put(blockName, new Ref(blockName, buf, q));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean evictBlock(String blockName) {
|
||||||
|
return cache.remove(blockName) != null;
|
||||||
|
}
|
||||||
|
|
||||||
public void shutdown() {
|
public void shutdown() {
|
||||||
// noop
|
// noop
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public CacheStats getStats() {
|
||||||
|
// TODO: implement this if we ever actually use this block cache
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -184,7 +184,7 @@ public class LoadIncrementalHFiles extends Configured implements Tool {
|
||||||
throws IOException {
|
throws IOException {
|
||||||
final Path hfilePath = item.hfilePath;
|
final Path hfilePath = item.hfilePath;
|
||||||
final FileSystem fs = hfilePath.getFileSystem(getConf());
|
final FileSystem fs = hfilePath.getFileSystem(getConf());
|
||||||
HFile.Reader hfr = new HFile.Reader(fs, hfilePath, null, false);
|
HFile.Reader hfr = new HFile.Reader(fs, hfilePath, null, false, false);
|
||||||
final byte[] first, last;
|
final byte[] first, last;
|
||||||
try {
|
try {
|
||||||
hfr.loadFileInfo();
|
hfr.loadFileInfo();
|
||||||
|
@ -276,7 +276,7 @@ public class LoadIncrementalHFiles extends Configured implements Tool {
|
||||||
|
|
||||||
halfWriter = new StoreFile.Writer(
|
halfWriter = new StoreFile.Writer(
|
||||||
fs, outFile, blocksize, compression, conf, KeyValue.COMPARATOR,
|
fs, outFile, blocksize, compression, conf, KeyValue.COMPARATOR,
|
||||||
bloomFilterType, 0);
|
bloomFilterType, 0, false);
|
||||||
HFileScanner scanner = halfReader.getScanner(false, false);
|
HFileScanner scanner = halfReader.getScanner(false, false);
|
||||||
scanner.seekTo();
|
scanner.seekTo();
|
||||||
do {
|
do {
|
||||||
|
|
|
@ -44,6 +44,7 @@ import org.apache.hadoop.hbase.KeyValue;
|
||||||
import org.apache.hadoop.hbase.RemoteExceptionHandler;
|
import org.apache.hadoop.hbase.RemoteExceptionHandler;
|
||||||
import org.apache.hadoop.hbase.client.Scan;
|
import org.apache.hadoop.hbase.client.Scan;
|
||||||
import org.apache.hadoop.hbase.io.HeapSize;
|
import org.apache.hadoop.hbase.io.HeapSize;
|
||||||
|
import org.apache.hadoop.hbase.io.hfile.BlockCache;
|
||||||
import org.apache.hadoop.hbase.io.hfile.Compression;
|
import org.apache.hadoop.hbase.io.hfile.Compression;
|
||||||
import org.apache.hadoop.hbase.io.hfile.HFile;
|
import org.apache.hadoop.hbase.io.hfile.HFile;
|
||||||
import org.apache.hadoop.hbase.io.hfile.HFileScanner;
|
import org.apache.hadoop.hbase.io.hfile.HFileScanner;
|
||||||
|
@ -324,7 +325,7 @@ public class Store implements HeapSize {
|
||||||
LOG.info("Validating hfile at " + srcPath + " for inclusion in "
|
LOG.info("Validating hfile at " + srcPath + " for inclusion in "
|
||||||
+ "store " + this + " region " + this.region);
|
+ "store " + this + " region " + this.region);
|
||||||
reader = new HFile.Reader(srcPath.getFileSystem(conf),
|
reader = new HFile.Reader(srcPath.getFileSystem(conf),
|
||||||
srcPath, null, false);
|
srcPath, null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
|
|
||||||
byte[] firstKey = reader.getFirstRowKey();
|
byte[] firstKey = reader.getFirstRowKey();
|
||||||
|
@ -527,7 +528,8 @@ public class Store implements HeapSize {
|
||||||
throws IOException {
|
throws IOException {
|
||||||
return StoreFile.createWriter(this.fs, region.getTmpDir(), this.blocksize,
|
return StoreFile.createWriter(this.fs, region.getTmpDir(), this.blocksize,
|
||||||
compression, this.comparator, this.conf,
|
compression, this.comparator, this.conf,
|
||||||
this.family.getBloomFilterType(), maxKeyCount);
|
this.family.getBloomFilterType(), maxKeyCount,
|
||||||
|
conf.getBoolean("hbase.rs.cacheblocksonwrite", false));
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -19,6 +19,23 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.hadoop.hbase.regionserver;
|
package org.apache.hadoop.hbase.regionserver;
|
||||||
|
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.lang.management.ManagementFactory;
|
||||||
|
import java.lang.management.MemoryUsage;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.text.NumberFormat;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.Comparator;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Random;
|
||||||
|
import java.util.SortedSet;
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
import java.util.regex.Matcher;
|
||||||
|
import java.util.regex.Pattern;
|
||||||
|
|
||||||
import org.apache.commons.logging.Log;
|
import org.apache.commons.logging.Log;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.apache.commons.logging.LogFactory;
|
||||||
import org.apache.hadoop.conf.Configuration;
|
import org.apache.hadoop.conf.Configuration;
|
||||||
|
@ -47,23 +64,6 @@ import com.google.common.base.Function;
|
||||||
import com.google.common.collect.ImmutableList;
|
import com.google.common.collect.ImmutableList;
|
||||||
import com.google.common.collect.Ordering;
|
import com.google.common.collect.Ordering;
|
||||||
|
|
||||||
import java.io.FileNotFoundException;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.lang.management.ManagementFactory;
|
|
||||||
import java.lang.management.MemoryUsage;
|
|
||||||
import java.nio.ByteBuffer;
|
|
||||||
import java.text.NumberFormat;
|
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.Comparator;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Random;
|
|
||||||
import java.util.SortedSet;
|
|
||||||
import java.util.concurrent.atomic.AtomicBoolean;
|
|
||||||
import java.util.regex.Matcher;
|
|
||||||
import java.util.regex.Pattern;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A Store data file. Stores usually have one or more of these files. They
|
* A Store data file. Stores usually have one or more of these files. They
|
||||||
* are produced by flushing the memstore to disk. To
|
* are produced by flushing the memstore to disk. To
|
||||||
|
@ -376,7 +376,8 @@ public class StoreFile {
|
||||||
getBlockCache(), this.reference);
|
getBlockCache(), this.reference);
|
||||||
} else {
|
} else {
|
||||||
this.reader = new Reader(this.fs, this.path, getBlockCache(),
|
this.reader = new Reader(this.fs, this.path, getBlockCache(),
|
||||||
this.inMemory);
|
this.inMemory,
|
||||||
|
this.conf.getBoolean("hbase.rs.evictblocksonclose", true));
|
||||||
}
|
}
|
||||||
// Load up indices and fileinfo.
|
// Load up indices and fileinfo.
|
||||||
metadataMap = Collections.unmodifiableMap(this.reader.loadFileInfo());
|
metadataMap = Collections.unmodifiableMap(this.reader.loadFileInfo());
|
||||||
|
@ -529,7 +530,8 @@ public class StoreFile {
|
||||||
final int blocksize)
|
final int blocksize)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
|
|
||||||
return createWriter(fs, dir, blocksize, null, null, null, BloomType.NONE, 0);
|
return createWriter(fs, dir, blocksize, null, null, null, BloomType.NONE, 0,
|
||||||
|
false);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -554,7 +556,8 @@ public class StoreFile {
|
||||||
final KeyValue.KVComparator c,
|
final KeyValue.KVComparator c,
|
||||||
final Configuration conf,
|
final Configuration conf,
|
||||||
BloomType bloomType,
|
BloomType bloomType,
|
||||||
int maxKeySize)
|
int maxKeySize,
|
||||||
|
final boolean cacheOnWrite)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
|
|
||||||
if (!fs.exists(dir)) {
|
if (!fs.exists(dir)) {
|
||||||
|
@ -567,7 +570,8 @@ public class StoreFile {
|
||||||
|
|
||||||
return new Writer(fs, path, blocksize,
|
return new Writer(fs, path, blocksize,
|
||||||
algorithm == null? HFile.DEFAULT_COMPRESSION_ALGORITHM: algorithm,
|
algorithm == null? HFile.DEFAULT_COMPRESSION_ALGORITHM: algorithm,
|
||||||
conf, c == null? KeyValue.COMPARATOR: c, bloomType, maxKeySize);
|
conf, c == null? KeyValue.COMPARATOR: c, bloomType, maxKeySize,
|
||||||
|
cacheOnWrite);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -682,13 +686,17 @@ public class StoreFile {
|
||||||
* @param comparator key comparator
|
* @param comparator key comparator
|
||||||
* @param bloomType bloom filter setting
|
* @param bloomType bloom filter setting
|
||||||
* @param maxKeys maximum amount of keys to add (for blooms)
|
* @param maxKeys maximum amount of keys to add (for blooms)
|
||||||
|
* @param cacheOnWrite whether to cache blocks as we write file
|
||||||
* @throws IOException problem writing to FS
|
* @throws IOException problem writing to FS
|
||||||
*/
|
*/
|
||||||
public Writer(FileSystem fs, Path path, int blocksize,
|
public Writer(FileSystem fs, Path path, int blocksize,
|
||||||
Compression.Algorithm compress, final Configuration conf,
|
Compression.Algorithm compress, final Configuration conf,
|
||||||
final KVComparator comparator, BloomType bloomType, int maxKeys)
|
final KVComparator comparator, BloomType bloomType, int maxKeys,
|
||||||
|
boolean cacheOnWrite)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
writer = new HFile.Writer(fs, path, blocksize, compress, comparator.getRawComparator());
|
writer = new HFile.Writer(fs, path, blocksize, compress,
|
||||||
|
comparator.getRawComparator(),
|
||||||
|
cacheOnWrite ? StoreFile.getBlockCache(conf) : null);
|
||||||
|
|
||||||
this.kvComparator = comparator;
|
this.kvComparator = comparator;
|
||||||
|
|
||||||
|
@ -894,9 +902,10 @@ public class StoreFile {
|
||||||
protected TimeRangeTracker timeRangeTracker = null;
|
protected TimeRangeTracker timeRangeTracker = null;
|
||||||
protected long sequenceID = -1;
|
protected long sequenceID = -1;
|
||||||
|
|
||||||
public Reader(FileSystem fs, Path path, BlockCache blockCache, boolean inMemory)
|
public Reader(FileSystem fs, Path path, BlockCache blockCache,
|
||||||
|
boolean inMemory, boolean evictOnClose)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
reader = new HFile.Reader(fs, path, blockCache, inMemory);
|
reader = new HFile.Reader(fs, path, blockCache, inMemory, evictOnClose);
|
||||||
bloomFilterType = BloomType.NONE;
|
bloomFilterType = BloomType.NONE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -127,7 +127,7 @@ public class CompressionTest {
|
||||||
writer.appendFileInfo(Bytes.toBytes("infokey"), Bytes.toBytes("infoval"));
|
writer.appendFileInfo(Bytes.toBytes("infokey"), Bytes.toBytes("infoval"));
|
||||||
writer.close();
|
writer.close();
|
||||||
|
|
||||||
HFile.Reader reader = new HFile.Reader(dfs, path, null, false);
|
HFile.Reader reader = new HFile.Reader(dfs, path, null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
byte[] key = reader.getFirstKey();
|
byte[] key = reader.getFirstKey();
|
||||||
boolean rc = Bytes.toString(key).equals("testkey");
|
boolean rc = Bytes.toString(key).equals("testkey");
|
||||||
|
|
|
@ -188,7 +188,8 @@ public class HFilePerformanceEvaluation {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void setUp() throws Exception {
|
void setUp() throws Exception {
|
||||||
writer = new HFile.Writer(this.fs, this.mf, RFILE_BLOCKSIZE, (Compression.Algorithm) null, null);
|
writer = new HFile.Writer(this.fs, this.mf, RFILE_BLOCKSIZE,
|
||||||
|
(Compression.Algorithm) null, null, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -224,7 +225,7 @@ public class HFilePerformanceEvaluation {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void setUp() throws Exception {
|
void setUp() throws Exception {
|
||||||
reader = new HFile.Reader(this.fs, this.mf, null, false);
|
reader = new HFile.Reader(this.fs, this.mf, null, false, false);
|
||||||
this.reader.loadFileInfo();
|
this.reader.loadFileInfo();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -71,7 +71,7 @@ public class TestHalfStoreFileReader {
|
||||||
}
|
}
|
||||||
w.close();
|
w.close();
|
||||||
|
|
||||||
HFile.Reader r = new HFile.Reader(fs, p, null, false);
|
HFile.Reader r = new HFile.Reader(fs, p, null, false, false);
|
||||||
r.loadFileInfo();
|
r.loadFileInfo();
|
||||||
byte [] midkey = r.midkey();
|
byte [] midkey = r.midkey();
|
||||||
KeyValue midKV = KeyValue.createKeyValueFromKey(midkey);
|
KeyValue midKV = KeyValue.createKeyValueFromKey(midkey);
|
||||||
|
|
|
@ -68,7 +68,7 @@ public class RandomSeek {
|
||||||
long start = System.currentTimeMillis();
|
long start = System.currentTimeMillis();
|
||||||
SimpleBlockCache cache = new SimpleBlockCache();
|
SimpleBlockCache cache = new SimpleBlockCache();
|
||||||
//LruBlockCache cache = new LruBlockCache();
|
//LruBlockCache cache = new LruBlockCache();
|
||||||
Reader reader = new HFile.Reader(lfs, path, cache, false);
|
Reader reader = new HFile.Reader(lfs, path, cache, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
System.out.println(reader.trailer);
|
System.out.println(reader.trailer);
|
||||||
long end = System.currentTimeMillis();
|
long end = System.currentTimeMillis();
|
||||||
|
|
|
@ -65,7 +65,7 @@ public class TestHFile extends HBaseTestCase {
|
||||||
Path f = new Path(ROOT_DIR, getName());
|
Path f = new Path(ROOT_DIR, getName());
|
||||||
Writer w = new Writer(this.fs, f);
|
Writer w = new Writer(this.fs, f);
|
||||||
w.close();
|
w.close();
|
||||||
Reader r = new Reader(fs, f, null, false);
|
Reader r = new Reader(fs, f, null, false, false);
|
||||||
r.loadFileInfo();
|
r.loadFileInfo();
|
||||||
assertNull(r.getFirstKey());
|
assertNull(r.getFirstKey());
|
||||||
assertNull(r.getLastKey());
|
assertNull(r.getLastKey());
|
||||||
|
@ -140,8 +140,8 @@ public class TestHFile extends HBaseTestCase {
|
||||||
writeRecords(writer);
|
writeRecords(writer);
|
||||||
fout.close();
|
fout.close();
|
||||||
FSDataInputStream fin = fs.open(ncTFile);
|
FSDataInputStream fin = fs.open(ncTFile);
|
||||||
Reader reader = new Reader(fs.open(ncTFile),
|
Reader reader = new Reader(ncTFile, fs.open(ncTFile),
|
||||||
fs.getFileStatus(ncTFile).getLen(), null, false);
|
fs.getFileStatus(ncTFile).getLen(), null, false, false);
|
||||||
// Load up the index.
|
// Load up the index.
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
// Get a scanner that caches and that does not use pread.
|
// Get a scanner that caches and that does not use pread.
|
||||||
|
@ -215,8 +215,8 @@ public class TestHFile extends HBaseTestCase {
|
||||||
writer.close();
|
writer.close();
|
||||||
fout.close();
|
fout.close();
|
||||||
FSDataInputStream fin = fs.open(mFile);
|
FSDataInputStream fin = fs.open(mFile);
|
||||||
Reader reader = new Reader(fs.open(mFile), this.fs.getFileStatus(mFile)
|
Reader reader = new Reader(mFile, fs.open(mFile),
|
||||||
.getLen(), null, false);
|
this.fs.getFileStatus(mFile).getLen(), null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
// No data -- this should return false.
|
// No data -- this should return false.
|
||||||
assertFalse(reader.getScanner(false, false).seekTo());
|
assertFalse(reader.getScanner(false, false).seekTo());
|
||||||
|
@ -240,7 +240,7 @@ public class TestHFile extends HBaseTestCase {
|
||||||
writer.append("foo".getBytes(), "value".getBytes());
|
writer.append("foo".getBytes(), "value".getBytes());
|
||||||
writer.close();
|
writer.close();
|
||||||
fout.close();
|
fout.close();
|
||||||
Reader reader = new Reader(fs, mFile, null, false);
|
Reader reader = new Reader(fs, mFile, null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
assertNull(reader.getMetaBlock("non-existant", false));
|
assertNull(reader.getMetaBlock("non-existant", false));
|
||||||
}
|
}
|
||||||
|
|
|
@ -236,8 +236,8 @@ public class TestHFilePerformance extends TestCase {
|
||||||
FSDataInputStream fin = fs.open(path);
|
FSDataInputStream fin = fs.open(path);
|
||||||
|
|
||||||
if ("HFile".equals(fileType)){
|
if ("HFile".equals(fileType)){
|
||||||
HFile.Reader reader = new HFile.Reader(fs.open(path),
|
HFile.Reader reader = new HFile.Reader(path, fs.open(path),
|
||||||
fs.getFileStatus(path).getLen(), null, false);
|
fs.getFileStatus(path).getLen(), null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
switch (method) {
|
switch (method) {
|
||||||
|
|
||||||
|
|
|
@ -155,8 +155,8 @@ public class TestHFileSeek extends TestCase {
|
||||||
int miss = 0;
|
int miss = 0;
|
||||||
long totalBytes = 0;
|
long totalBytes = 0;
|
||||||
FSDataInputStream fsdis = fs.open(path);
|
FSDataInputStream fsdis = fs.open(path);
|
||||||
Reader reader =
|
Reader reader = new Reader(path, fsdis, fs.getFileStatus(path).getLen(),
|
||||||
new Reader(fsdis, fs.getFileStatus(path).getLen(), null, false);
|
null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
KeySampler kSampler =
|
KeySampler kSampler =
|
||||||
new KeySampler(rng, reader.getFirstKey(), reader.getLastKey(),
|
new KeySampler(rng, reader.getFirstKey(), reader.getLastKey(),
|
||||||
|
|
|
@ -60,7 +60,7 @@ public class TestReseekTo {
|
||||||
fout.close();
|
fout.close();
|
||||||
|
|
||||||
HFile.Reader reader = new HFile.Reader(TEST_UTIL.getTestFileSystem(),
|
HFile.Reader reader = new HFile.Reader(TEST_UTIL.getTestFileSystem(),
|
||||||
ncTFile, null, false);
|
ncTFile, null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
HFileScanner scanner = reader.getScanner(false, true);
|
HFileScanner scanner = reader.getScanner(false, true);
|
||||||
|
|
||||||
|
|
|
@ -49,7 +49,7 @@ public class TestSeekTo extends HBaseTestCase {
|
||||||
}
|
}
|
||||||
public void testSeekBefore() throws Exception {
|
public void testSeekBefore() throws Exception {
|
||||||
Path p = makeNewFile();
|
Path p = makeNewFile();
|
||||||
HFile.Reader reader = new HFile.Reader(fs, p, null, false);
|
HFile.Reader reader = new HFile.Reader(fs, p, null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
HFileScanner scanner = reader.getScanner(false, true);
|
HFileScanner scanner = reader.getScanner(false, true);
|
||||||
assertEquals(false, scanner.seekBefore(Bytes.toBytes("a")));
|
assertEquals(false, scanner.seekBefore(Bytes.toBytes("a")));
|
||||||
|
@ -82,7 +82,7 @@ public class TestSeekTo extends HBaseTestCase {
|
||||||
|
|
||||||
public void testSeekTo() throws Exception {
|
public void testSeekTo() throws Exception {
|
||||||
Path p = makeNewFile();
|
Path p = makeNewFile();
|
||||||
HFile.Reader reader = new HFile.Reader(fs, p, null, false);
|
HFile.Reader reader = new HFile.Reader(fs, p, null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
assertEquals(2, reader.blockIndex.count);
|
assertEquals(2, reader.blockIndex.count);
|
||||||
HFileScanner scanner = reader.getScanner(false, true);
|
HFileScanner scanner = reader.getScanner(false, true);
|
||||||
|
@ -102,7 +102,7 @@ public class TestSeekTo extends HBaseTestCase {
|
||||||
|
|
||||||
public void testBlockContainingKey() throws Exception {
|
public void testBlockContainingKey() throws Exception {
|
||||||
Path p = makeNewFile();
|
Path p = makeNewFile();
|
||||||
HFile.Reader reader = new HFile.Reader(fs, p, null, false);
|
HFile.Reader reader = new HFile.Reader(fs, p, null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
System.out.println(reader.blockIndex.toString());
|
System.out.println(reader.blockIndex.toString());
|
||||||
// falls before the start of the file.
|
// falls before the start of the file.
|
||||||
|
|
|
@ -150,7 +150,7 @@ public class TestLoadIncrementalHFiles {
|
||||||
private int verifyHFile(Path p) throws IOException {
|
private int verifyHFile(Path p) throws IOException {
|
||||||
Configuration conf = util.getConfiguration();
|
Configuration conf = util.getConfiguration();
|
||||||
HFile.Reader reader = new HFile.Reader(
|
HFile.Reader reader = new HFile.Reader(
|
||||||
p.getFileSystem(conf), p, null, false);
|
p.getFileSystem(conf), p, null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
HFileScanner scanner = reader.getScanner(false, false);
|
HFileScanner scanner = reader.getScanner(false, false);
|
||||||
scanner.seekTo();
|
scanner.seekTo();
|
||||||
|
|
|
@ -30,6 +30,7 @@ import java.util.TreeSet;
|
||||||
|
|
||||||
import org.apache.commons.logging.Log;
|
import org.apache.commons.logging.Log;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.apache.commons.logging.LogFactory;
|
||||||
|
import org.apache.hadoop.conf.Configuration;
|
||||||
import org.apache.hadoop.fs.FileSystem;
|
import org.apache.hadoop.fs.FileSystem;
|
||||||
import org.apache.hadoop.fs.Path;
|
import org.apache.hadoop.fs.Path;
|
||||||
import org.apache.hadoop.hbase.HBaseTestCase;
|
import org.apache.hadoop.hbase.HBaseTestCase;
|
||||||
|
@ -38,16 +39,15 @@ import org.apache.hadoop.hbase.HConstants;
|
||||||
import org.apache.hadoop.hbase.KeyValue;
|
import org.apache.hadoop.hbase.KeyValue;
|
||||||
import org.apache.hadoop.hbase.client.Scan;
|
import org.apache.hadoop.hbase.client.Scan;
|
||||||
import org.apache.hadoop.hbase.io.Reference.Range;
|
import org.apache.hadoop.hbase.io.Reference.Range;
|
||||||
|
import org.apache.hadoop.hbase.io.hfile.BlockCache;
|
||||||
import org.apache.hadoop.hbase.io.hfile.HFile;
|
import org.apache.hadoop.hbase.io.hfile.HFile;
|
||||||
import org.apache.hadoop.hbase.io.hfile.HFileScanner;
|
import org.apache.hadoop.hbase.io.hfile.HFileScanner;
|
||||||
import org.apache.hadoop.hbase.util.ByteBloomFilter;
|
import org.apache.hadoop.hbase.io.hfile.LruBlockCache.CacheStats;
|
||||||
import org.apache.hadoop.hbase.util.Bytes;
|
import org.apache.hadoop.hbase.util.Bytes;
|
||||||
import org.apache.hadoop.hbase.util.Hash;
|
|
||||||
import org.apache.hadoop.hdfs.MiniDFSCluster;
|
import org.apache.hadoop.hdfs.MiniDFSCluster;
|
||||||
import org.mockito.Mockito;
|
import org.mockito.Mockito;
|
||||||
|
|
||||||
import com.google.common.base.Joiner;
|
import com.google.common.base.Joiner;
|
||||||
import com.google.common.collect.Collections2;
|
|
||||||
import com.google.common.collect.Iterables;
|
import com.google.common.collect.Iterables;
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
|
|
||||||
|
@ -338,7 +338,7 @@ public class TestStoreFile extends HBaseTestCase {
|
||||||
}
|
}
|
||||||
writer.close();
|
writer.close();
|
||||||
|
|
||||||
StoreFile.Reader reader = new StoreFile.Reader(fs, f, null, false);
|
StoreFile.Reader reader = new StoreFile.Reader(fs, f, null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
reader.loadBloomfilter();
|
reader.loadBloomfilter();
|
||||||
StoreFileScanner scanner = reader.getStoreFileScanner(false, false);
|
StoreFileScanner scanner = reader.getStoreFileScanner(false, false);
|
||||||
|
@ -380,7 +380,7 @@ public class TestStoreFile extends HBaseTestCase {
|
||||||
Path f = new Path(ROOT_DIR, getName());
|
Path f = new Path(ROOT_DIR, getName());
|
||||||
StoreFile.Writer writer = new StoreFile.Writer(fs, f,
|
StoreFile.Writer writer = new StoreFile.Writer(fs, f,
|
||||||
StoreFile.DEFAULT_BLOCKSIZE_SMALL, HFile.DEFAULT_COMPRESSION_ALGORITHM,
|
StoreFile.DEFAULT_BLOCKSIZE_SMALL, HFile.DEFAULT_COMPRESSION_ALGORITHM,
|
||||||
conf, KeyValue.COMPARATOR, StoreFile.BloomType.ROW, 2000);
|
conf, KeyValue.COMPARATOR, StoreFile.BloomType.ROW, 2000, false);
|
||||||
|
|
||||||
bloomWriteRead(writer, fs);
|
bloomWriteRead(writer, fs);
|
||||||
}
|
}
|
||||||
|
@ -411,7 +411,7 @@ public class TestStoreFile extends HBaseTestCase {
|
||||||
StoreFile.Writer writer = new StoreFile.Writer(fs, f,
|
StoreFile.Writer writer = new StoreFile.Writer(fs, f,
|
||||||
StoreFile.DEFAULT_BLOCKSIZE_SMALL,
|
StoreFile.DEFAULT_BLOCKSIZE_SMALL,
|
||||||
HFile.DEFAULT_COMPRESSION_ALGORITHM,
|
HFile.DEFAULT_COMPRESSION_ALGORITHM,
|
||||||
conf, KeyValue.COMPARATOR, bt[x], expKeys[x]);
|
conf, KeyValue.COMPARATOR, bt[x], expKeys[x], false);
|
||||||
|
|
||||||
long now = System.currentTimeMillis();
|
long now = System.currentTimeMillis();
|
||||||
for (int i = 0; i < rowCount*2; i += 2) { // rows
|
for (int i = 0; i < rowCount*2; i += 2) { // rows
|
||||||
|
@ -428,7 +428,7 @@ public class TestStoreFile extends HBaseTestCase {
|
||||||
}
|
}
|
||||||
writer.close();
|
writer.close();
|
||||||
|
|
||||||
StoreFile.Reader reader = new StoreFile.Reader(fs, f, null, false);
|
StoreFile.Reader reader = new StoreFile.Reader(fs, f, null, false, false);
|
||||||
reader.loadFileInfo();
|
reader.loadFileInfo();
|
||||||
reader.loadBloomfilter();
|
reader.loadBloomfilter();
|
||||||
StoreFileScanner scanner = reader.getStoreFileScanner(false, false);
|
StoreFileScanner scanner = reader.getStoreFileScanner(false, false);
|
||||||
|
@ -478,7 +478,7 @@ public class TestStoreFile extends HBaseTestCase {
|
||||||
// this should not create a bloom because the max keys is too small
|
// this should not create a bloom because the max keys is too small
|
||||||
StoreFile.Writer writer = new StoreFile.Writer(fs, f,
|
StoreFile.Writer writer = new StoreFile.Writer(fs, f,
|
||||||
StoreFile.DEFAULT_BLOCKSIZE_SMALL, HFile.DEFAULT_COMPRESSION_ALGORITHM,
|
StoreFile.DEFAULT_BLOCKSIZE_SMALL, HFile.DEFAULT_COMPRESSION_ALGORITHM,
|
||||||
conf, KeyValue.COMPARATOR, StoreFile.BloomType.ROW, 2000);
|
conf, KeyValue.COMPARATOR, StoreFile.BloomType.ROW, 2000, false);
|
||||||
assertFalse(writer.hasBloom());
|
assertFalse(writer.hasBloom());
|
||||||
writer.close();
|
writer.close();
|
||||||
fs.delete(f, true);
|
fs.delete(f, true);
|
||||||
|
@ -500,7 +500,8 @@ public class TestStoreFile extends HBaseTestCase {
|
||||||
// because Java can't create a contiguous array > MAX_INT
|
// because Java can't create a contiguous array > MAX_INT
|
||||||
writer = new StoreFile.Writer(fs, f,
|
writer = new StoreFile.Writer(fs, f,
|
||||||
StoreFile.DEFAULT_BLOCKSIZE_SMALL, HFile.DEFAULT_COMPRESSION_ALGORITHM,
|
StoreFile.DEFAULT_BLOCKSIZE_SMALL, HFile.DEFAULT_COMPRESSION_ALGORITHM,
|
||||||
conf, KeyValue.COMPARATOR, StoreFile.BloomType.ROW, Integer.MAX_VALUE);
|
conf, KeyValue.COMPARATOR, StoreFile.BloomType.ROW, Integer.MAX_VALUE,
|
||||||
|
false);
|
||||||
assertFalse(writer.hasBloom());
|
assertFalse(writer.hasBloom());
|
||||||
writer.close();
|
writer.close();
|
||||||
fs.delete(f, true);
|
fs.delete(f, true);
|
||||||
|
@ -626,4 +627,144 @@ public class TestStoreFile extends HBaseTestCase {
|
||||||
//scan.setTimeRange(27, 50);
|
//scan.setTimeRange(27, 50);
|
||||||
//assertTrue(!scanner.shouldSeek(scan, columns));
|
//assertTrue(!scanner.shouldSeek(scan, columns));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void testCacheOnWriteEvictOnClose() throws Exception {
|
||||||
|
Configuration conf = this.conf;
|
||||||
|
conf.setBoolean("hbase.rs.evictblocksonclose", false);
|
||||||
|
|
||||||
|
// Find a home for our files
|
||||||
|
Path baseDir = new Path(new Path(this.testDir, "regionname"),
|
||||||
|
"twoCOWEOC");
|
||||||
|
|
||||||
|
// Grab the block cache and get the initial hit/miss counts
|
||||||
|
BlockCache bc = StoreFile.getBlockCache(conf);
|
||||||
|
assertNotNull(bc);
|
||||||
|
CacheStats cs = bc.getStats();
|
||||||
|
long startHit = cs.getHitCount();
|
||||||
|
long startMiss = cs.getMissCount();
|
||||||
|
long startEvicted = cs.getEvictedCount();
|
||||||
|
|
||||||
|
// Let's write a StoreFile with three blocks, with cache on write off
|
||||||
|
conf.setBoolean("hbase.rs.cacheblocksonwrite", false);
|
||||||
|
Path pathCowOff = new Path(baseDir, "123456789");
|
||||||
|
StoreFile.Writer writer = writeStoreFile(conf, pathCowOff, 3);
|
||||||
|
StoreFile hsf = new StoreFile(this.fs, writer.getPath(), true, conf,
|
||||||
|
StoreFile.BloomType.NONE, false);
|
||||||
|
LOG.debug(hsf.getPath().toString());
|
||||||
|
|
||||||
|
// Read this file, we should see 3 misses
|
||||||
|
StoreFile.Reader reader = hsf.createReader();
|
||||||
|
reader.loadFileInfo();
|
||||||
|
StoreFileScanner scanner = reader.getStoreFileScanner(true, true);
|
||||||
|
scanner.seek(KeyValue.LOWESTKEY);
|
||||||
|
while (scanner.next() != null);
|
||||||
|
assertEquals(startHit, cs.getHitCount());
|
||||||
|
assertEquals(startMiss + 3, cs.getMissCount());
|
||||||
|
assertEquals(startEvicted, cs.getEvictedCount());
|
||||||
|
startMiss += 3;
|
||||||
|
scanner.close();
|
||||||
|
reader.close();
|
||||||
|
|
||||||
|
// Now write a StoreFile with three blocks, with cache on write on
|
||||||
|
conf.setBoolean("hbase.rs.cacheblocksonwrite", true);
|
||||||
|
Path pathCowOn = new Path(baseDir, "123456788");
|
||||||
|
writer = writeStoreFile(conf, pathCowOn, 3);
|
||||||
|
hsf = new StoreFile(this.fs, writer.getPath(), true, conf,
|
||||||
|
StoreFile.BloomType.NONE, false);
|
||||||
|
|
||||||
|
// Read this file, we should see 3 hits
|
||||||
|
reader = hsf.createReader();
|
||||||
|
scanner = reader.getStoreFileScanner(true, true);
|
||||||
|
scanner.seek(KeyValue.LOWESTKEY);
|
||||||
|
while (scanner.next() != null);
|
||||||
|
assertEquals(startHit + 3, cs.getHitCount());
|
||||||
|
assertEquals(startMiss, cs.getMissCount());
|
||||||
|
assertEquals(startEvicted, cs.getEvictedCount());
|
||||||
|
startHit += 3;
|
||||||
|
scanner.close();
|
||||||
|
reader.close();
|
||||||
|
|
||||||
|
// Let's read back the two files to ensure the blocks exactly match
|
||||||
|
hsf = new StoreFile(this.fs, pathCowOff, true, conf,
|
||||||
|
StoreFile.BloomType.NONE, false);
|
||||||
|
StoreFile.Reader readerOne = hsf.createReader();
|
||||||
|
readerOne.loadFileInfo();
|
||||||
|
StoreFileScanner scannerOne = readerOne.getStoreFileScanner(true, true);
|
||||||
|
scannerOne.seek(KeyValue.LOWESTKEY);
|
||||||
|
hsf = new StoreFile(this.fs, pathCowOn, true, conf,
|
||||||
|
StoreFile.BloomType.NONE, false);
|
||||||
|
StoreFile.Reader readerTwo = hsf.createReader();
|
||||||
|
readerTwo.loadFileInfo();
|
||||||
|
StoreFileScanner scannerTwo = readerTwo.getStoreFileScanner(true, true);
|
||||||
|
scannerTwo.seek(KeyValue.LOWESTKEY);
|
||||||
|
KeyValue kv1 = null;
|
||||||
|
KeyValue kv2 = null;
|
||||||
|
while ((kv1 = scannerOne.next()) != null) {
|
||||||
|
kv2 = scannerTwo.next();
|
||||||
|
assertTrue(kv1.equals(kv2));
|
||||||
|
assertTrue(Bytes.equals(kv1.getBuffer(), kv2.getBuffer()));
|
||||||
|
}
|
||||||
|
assertNull(scannerTwo.next());
|
||||||
|
assertEquals(startHit + 6, cs.getHitCount());
|
||||||
|
assertEquals(startMiss, cs.getMissCount());
|
||||||
|
assertEquals(startEvicted, cs.getEvictedCount());
|
||||||
|
startHit += 6;
|
||||||
|
scannerOne.close();
|
||||||
|
readerOne.close();
|
||||||
|
scannerTwo.close();
|
||||||
|
readerTwo.close();
|
||||||
|
|
||||||
|
// Let's close the first file with evict on close turned on
|
||||||
|
conf.setBoolean("hbase.rs.evictblocksonclose", true);
|
||||||
|
hsf = new StoreFile(this.fs, pathCowOff, true, conf,
|
||||||
|
StoreFile.BloomType.NONE, false);
|
||||||
|
reader = hsf.createReader();
|
||||||
|
reader.close();
|
||||||
|
|
||||||
|
// We should have 3 new evictions
|
||||||
|
assertEquals(startHit, cs.getHitCount());
|
||||||
|
assertEquals(startMiss, cs.getMissCount());
|
||||||
|
assertEquals(startEvicted + 3, cs.getEvictedCount());
|
||||||
|
startEvicted += 3;
|
||||||
|
|
||||||
|
// Let's close the second file with evict on close turned off
|
||||||
|
conf.setBoolean("hbase.rs.evictblocksonclose", false);
|
||||||
|
hsf = new StoreFile(this.fs, pathCowOn, true, conf,
|
||||||
|
StoreFile.BloomType.NONE, false);
|
||||||
|
reader = hsf.createReader();
|
||||||
|
reader.close();
|
||||||
|
|
||||||
|
// We expect no changes
|
||||||
|
assertEquals(startHit, cs.getHitCount());
|
||||||
|
assertEquals(startMiss, cs.getMissCount());
|
||||||
|
assertEquals(startEvicted, cs.getEvictedCount());
|
||||||
|
}
|
||||||
|
|
||||||
|
private StoreFile.Writer writeStoreFile(Configuration conf, Path path,
|
||||||
|
int numBlocks)
|
||||||
|
throws IOException {
|
||||||
|
// Let's put ~5 small KVs in each block, so let's make 5*numBlocks KVs
|
||||||
|
int numKVs = 5 * numBlocks;
|
||||||
|
List<KeyValue> kvs = new ArrayList<KeyValue>(numKVs);
|
||||||
|
byte [] b = Bytes.toBytes("x");
|
||||||
|
int totalSize = 0;
|
||||||
|
for (int i=numKVs;i>0;i--) {
|
||||||
|
KeyValue kv = new KeyValue(b, b, b, i, b);
|
||||||
|
kvs.add(kv);
|
||||||
|
totalSize += kv.getLength();
|
||||||
|
}
|
||||||
|
int blockSize = totalSize / numBlocks;
|
||||||
|
StoreFile.Writer writer = new StoreFile.Writer(fs, path, blockSize,
|
||||||
|
HFile.DEFAULT_COMPRESSION_ALGORITHM,
|
||||||
|
conf, KeyValue.COMPARATOR, StoreFile.BloomType.NONE, 2000,
|
||||||
|
conf.getBoolean("hbase.rs.cacheblocksonwrite", false));
|
||||||
|
// We'll write N-1 KVs to ensure we don't write an extra block
|
||||||
|
kvs.remove(kvs.size()-1);
|
||||||
|
for (KeyValue kv : kvs) {
|
||||||
|
writer.append(kv);
|
||||||
|
}
|
||||||
|
writer.appendMetadata(0, false);
|
||||||
|
writer.close();
|
||||||
|
return writer;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue