@ -30,13 +30,24 @@ class MockedBlockBasedTable : public BlockBasedTable {
virtual CachableEntry < FilterBlockReader > GetFilter (
virtual CachableEntry < FilterBlockReader > GetFilter (
FilePrefetchBuffer * , const BlockHandle & filter_blk_handle ,
FilePrefetchBuffer * , const BlockHandle & filter_blk_handle ,
const bool /* unused */ , bool /* unused */ , GetContext * /* unused */ ,
const bool /* unused */ , bool /* unused */ , GetContext * /* unused */ ,
const SliceTransform * /* unused */ ) const override {
const SliceTransform * prefix_extractor ) const override {
Slice slice = slices [ filter_blk_handle . offset ( ) ] ;
Slice slice = slices [ filter_blk_handle . offset ( ) ] ;
auto obj = new FullFilterBlockReader (
auto obj = new FullFilterBlockReader (
nullpt r, true , BlockContents ( slice , false , kNoCompression ) ,
prefix_extracto r, true , BlockContents ( slice , false , kNoCompression ) ,
rep_ - > table_options . filter_policy - > GetFilterBitsReader ( slice ) , nullptr ) ;
rep_ - > table_options . filter_policy - > GetFilterBitsReader ( slice ) , nullptr ) ;
return { obj , nullptr } ;
return { obj , nullptr } ;
}
}
virtual FilterBlockReader * ReadFilter (
FilePrefetchBuffer * , const BlockHandle & filter_blk_handle ,
const bool /* unused */ ,
const SliceTransform * prefix_extractor ) const override {
Slice slice = slices [ filter_blk_handle . offset ( ) ] ;
auto obj = new FullFilterBlockReader (
prefix_extractor , true , BlockContents ( slice , false , kNoCompression ) ,
rep_ - > table_options . filter_policy - > GetFilterBitsReader ( slice ) , nullptr ) ;
return obj ;
}
} ;
} ;
class PartitionedFilterBlockTest : public testing : : Test {
class PartitionedFilterBlockTest : public testing : : Test {
@ -93,7 +104,8 @@ class PartitionedFilterBlockTest : public testing::Test {
}
}
PartitionedFilterBlockBuilder * NewBuilder (
PartitionedFilterBlockBuilder * NewBuilder (
PartitionedIndexBuilder * const p_index_builder ) {
PartitionedIndexBuilder * const p_index_builder ,
const SliceTransform * prefix_extractor = nullptr ) {
assert ( table_options_ . block_size_deviation < = 100 ) ;
assert ( table_options_ . block_size_deviation < = 100 ) ;
auto partition_size = static_cast < uint32_t > (
auto partition_size = static_cast < uint32_t > (
( ( table_options_ . metadata_block_size *
( ( table_options_ . metadata_block_size *
@ -102,7 +114,7 @@ class PartitionedFilterBlockTest : public testing::Test {
100 ) ;
100 ) ;
partition_size = std : : max ( partition_size , static_cast < uint32_t > ( 1 ) ) ;
partition_size = std : : max ( partition_size , static_cast < uint32_t > ( 1 ) ) ;
return new PartitionedFilterBlockBuilder (
return new PartitionedFilterBlockBuilder (
nullpt r, table_options_ . whole_key_filtering ,
prefix_extracto r, table_options_ . whole_key_filtering ,
table_options_ . filter_policy - > GetFilterBitsBuilder ( ) ,
table_options_ . filter_policy - > GetFilterBitsBuilder ( ) ,
table_options_ . index_block_restart_interval , p_index_builder ,
table_options_ . index_block_restart_interval , p_index_builder ,
partition_size ) ;
partition_size ) ;
@ -111,7 +123,8 @@ class PartitionedFilterBlockTest : public testing::Test {
std : : unique_ptr < MockedBlockBasedTable > table ;
std : : unique_ptr < MockedBlockBasedTable > table ;
PartitionedFilterBlockReader * NewReader (
PartitionedFilterBlockReader * NewReader (
PartitionedFilterBlockBuilder * builder , PartitionedIndexBuilder * pib ) {
PartitionedFilterBlockBuilder * builder , PartitionedIndexBuilder * pib ,
const SliceTransform * prefix_extractor ) {
BlockHandle bh ;
BlockHandle bh ;
Status status ;
Status status ;
Slice slice ;
Slice slice ;
@ -126,41 +139,42 @@ class PartitionedFilterBlockTest : public testing::Test {
table . reset ( new MockedBlockBasedTable ( new BlockBasedTable : : Rep (
table . reset ( new MockedBlockBasedTable ( new BlockBasedTable : : Rep (
ioptions , env_options , table_options_ , icomp , false ) ) ) ;
ioptions , env_options , table_options_ , icomp , false ) ) ) ;
auto reader = new PartitionedFilterBlockReader (
auto reader = new PartitionedFilterBlockReader (
nullpt r, true , BlockContents ( slice , false , kNoCompression ) , nullptr ,
prefix_extracto r, true , BlockContents ( slice , false , kNoCompression ) ,
nullptr , icomp , table . get ( ) , pib - > seperator_is_key_plus_seq ( ) ) ;
nullptr , nullptr , icomp , table . get ( ) , pib - > seperator_is_key_plus_seq ( ) ) ;
return reader ;
return reader ;
}
}
void VerifyReader ( PartitionedFilterBlockBuilder * builder ,
void VerifyReader ( PartitionedFilterBlockBuilder * builder ,
PartitionedIndexBuilder * pib , bool empty = false ) {
PartitionedIndexBuilder * pib , bool empty = false ,
const SliceTransform * prefix_extractor = nullptr ) {
std : : unique_ptr < PartitionedFilterBlockReader > reader (
std : : unique_ptr < PartitionedFilterBlockReader > reader (
NewReader ( builder , pib ) ) ;
NewReader ( builder , pib , prefix_extractor ) ) ;
// Querying added keys
// Querying added keys
const bool no_io = true ;
const bool no_io = true ;
for ( auto key : keys ) {
for ( auto key : keys ) {
auto ikey = InternalKey ( key , 0 , ValueType : : kTypeValue ) ;
auto ikey = InternalKey ( key , 0 , ValueType : : kTypeValue ) ;
const Slice ikey_slice = Slice ( * ikey . rep ( ) ) ;
const Slice ikey_slice = Slice ( * ikey . rep ( ) ) ;
ASSERT_TRUE (
ASSERT_TRUE ( reader - > KeyMayMatch ( key , prefix_extractor , kNotValid , ! no_io ,
reader - > KeyMayMatch ( key , nullptr , kNotValid , ! no_io , & ikey_slice ) ) ;
& ikey_slice ) ) ;
}
}
{
{
// querying a key twice
// querying a key twice
auto ikey = InternalKey ( keys [ 0 ] , 0 , ValueType : : kTypeValue ) ;
auto ikey = InternalKey ( keys [ 0 ] , 0 , ValueType : : kTypeValue ) ;
const Slice ikey_slice = Slice ( * ikey . rep ( ) ) ;
const Slice ikey_slice = Slice ( * ikey . rep ( ) ) ;
ASSERT_TRUE ( reader - > KeyMayMatch ( keys [ 0 ] , nullpt r, kNotValid , ! no_io ,
ASSERT_TRUE ( reader - > KeyMayMatch ( keys [ 0 ] , prefix_extracto r, kNotValid ,
& ikey_slice ) ) ;
! no_io , & ikey_slice ) ) ;
}
}
// querying missing keys
// querying missing keys
for ( auto key : missing_keys ) {
for ( auto key : missing_keys ) {
auto ikey = InternalKey ( key , 0 , ValueType : : kTypeValue ) ;
auto ikey = InternalKey ( key , 0 , ValueType : : kTypeValue ) ;
const Slice ikey_slice = Slice ( * ikey . rep ( ) ) ;
const Slice ikey_slice = Slice ( * ikey . rep ( ) ) ;
if ( empty ) {
if ( empty ) {
ASSERT_TRUE (
ASSERT_TRUE ( reader - > KeyMayMatch ( key , prefix_extractor , kNotValid ,
reader - > KeyMayMatch ( key , nullptr , kNotValid , ! no_io , & ikey_slice ) ) ;
! no_io , & ikey_slice ) ) ;
} else {
} else {
// assuming a good hash function
// assuming a good hash function
ASSERT_FALSE (
ASSERT_FALSE ( reader - > KeyMayMatch ( key , prefix_extractor , kNotValid ,
reader - > KeyMayMatch ( key , nullptr , kNotValid , ! no_io , & ikey_slice ) ) ;
! no_io , & ikey_slice ) ) ;
}
}
}
}
}
}
@ -187,10 +201,10 @@ class PartitionedFilterBlockTest : public testing::Test {
return CountNumOfIndexPartitions ( pib . get ( ) ) ;
return CountNumOfIndexPartitions ( pib . get ( ) ) ;
}
}
void TestBlockPerTwoKeys ( ) {
void TestBlockPerTwoKeys ( const SliceTransform * prefix_extractor = nullptr ) {
std : : unique_ptr < PartitionedIndexBuilder > pib ( NewIndexBuilder ( ) ) ;
std : : unique_ptr < PartitionedIndexBuilder > pib ( NewIndexBuilder ( ) ) ;
std : : unique_ptr < PartitionedFilterBlockBuilder > builder (
std : : unique_ptr < PartitionedFilterBlockBuilder > builder (
NewBuilder ( pib . get ( ) ) ) ;
NewBuilder ( pib . get ( ) , prefix_extractor ) ) ;
int i = 0 ;
int i = 0 ;
builder - > Add ( keys [ i ] ) ;
builder - > Add ( keys [ i ] ) ;
i + + ;
i + + ;
@ -203,7 +217,7 @@ class PartitionedFilterBlockTest : public testing::Test {
builder - > Add ( keys [ i ] ) ;
builder - > Add ( keys [ i ] ) ;
CutABlock ( pib . get ( ) , keys [ i ] ) ;
CutABlock ( pib . get ( ) , keys [ i ] ) ;
VerifyReader ( builder . get ( ) , pib . get ( ) ) ;
VerifyReader ( builder . get ( ) , pib . get ( ) , prefix_extractor ) ;
}
}
void TestBlockPerAllKeys ( ) {
void TestBlockPerAllKeys ( ) {
@ -281,6 +295,34 @@ TEST_F(PartitionedFilterBlockTest, TwoBlocksPerKey) {
}
}
}
}
// This reproduces the bug that a prefix is the same among multiple consecutive
// blocks but the bug would add it only to the first block.
TEST_F ( PartitionedFilterBlockTest , SamePrefixInMultipleBlocks ) {
// some small number to cause partition cuts
table_options_ . metadata_block_size = 1 ;
std : : unique_ptr < const SliceTransform > prefix_extractor
( rocksdb : : NewFixedPrefixTransform ( 1 ) ) ;
std : : unique_ptr < PartitionedIndexBuilder > pib ( NewIndexBuilder ( ) ) ;
std : : unique_ptr < PartitionedFilterBlockBuilder > builder (
NewBuilder ( pib . get ( ) , prefix_extractor . get ( ) ) ) ;
const std : : string pkeys [ 3 ] = { " p-key1 " , " p-key2 " , " p-key3 " } ;
builder - > Add ( pkeys [ 0 ] ) ;
CutABlock ( pib . get ( ) , pkeys [ 0 ] , pkeys [ 1 ] ) ;
builder - > Add ( pkeys [ 1 ] ) ;
CutABlock ( pib . get ( ) , pkeys [ 1 ] , pkeys [ 2 ] ) ;
builder - > Add ( pkeys [ 2 ] ) ;
CutABlock ( pib . get ( ) , pkeys [ 2 ] ) ;
std : : unique_ptr < PartitionedFilterBlockReader > reader (
NewReader ( builder . get ( ) , pib . get ( ) , prefix_extractor . get ( ) ) ) ;
for ( auto key : pkeys ) {
auto ikey = InternalKey ( key , 0 , ValueType : : kTypeValue ) ;
const Slice ikey_slice = Slice ( * ikey . rep ( ) ) ;
ASSERT_TRUE ( reader - > PrefixMayMatch ( prefix_extractor - > Transform ( key ) ,
prefix_extractor . get ( ) , kNotValid ,
false /*no_io*/ , & ikey_slice ) ) ;
}
}
TEST_F ( PartitionedFilterBlockTest , OneBlockPerKey ) {
TEST_F ( PartitionedFilterBlockTest , OneBlockPerKey ) {
uint64_t max_index_size = MaxIndexSize ( ) ;
uint64_t max_index_size = MaxIndexSize ( ) ;
for ( uint64_t i = 1 ; i < max_index_size + 1 ; i + + ) {
for ( uint64_t i = 1 ; i < max_index_size + 1 ; i + + ) {