@ -13,6 +13,7 @@ 
			
		
	
		
		
			
				
					
					# include  <vector> # include  <vector>  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					# include  "db/arena_wrapped_db_iter.h" # include  "db/arena_wrapped_db_iter.h"  
			
		
	
		
		
			
				
					
					# include  "db/blob/blob_index.h"  
			
		
	
		
		
			
				
					
					# include  "db/column_family.h" # include  "db/column_family.h"  
			
		
	
		
		
			
				
					
					# include  "db/db_iter.h" # include  "db/db_iter.h"  
			
		
	
		
		
			
				
					
					# include  "db/db_test_util.h" # include  "db/db_test_util.h"  
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -138,20 +139,39 @@ class DBBlobIndexTest : public DBTestBase { 
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					} ; } ;  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					// Should be able to write kTypeBlobIndex to memtables and SST files.
 // Note: the following test case pertains to the StackableDB-based BlobDB
  
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					// implementation. We should be able to write kTypeBlobIndex to memtables and
  
			
		
	
		
		
			
				
					
					// SST files.
  
			
		
	
		
		
			
				
					
					TEST_F ( DBBlobIndexTest ,  Write )  { TEST_F ( DBBlobIndexTest ,  Write )  {  
			
		
	
		
		
			
				
					
					  for  ( auto  tier  :  kAllTiers )  {    for  ( auto  tier  :  kAllTiers )  {   
			
		
	
		
		
			
				
					
					    DestroyAndReopen ( GetTestOptions ( ) ) ;      DestroyAndReopen ( GetTestOptions ( ) ) ;   
			
		
	
		
		
			
				
					
					    for  ( int  i  =  1 ;  i  < =  5 ;  i + + )  {  
 
			
				
				
			
		
	
		
		
			
				
					
					      std : : string  index  =  ToString ( i ) ;      std : : vector < std : : pair < std : : string ,  std : : string > >  key_values ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    constexpr  size_t  num_key_values  =  5 ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    key_values . reserve ( num_key_values ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    for  ( size_t  i  =  1 ;  i  < =  num_key_values ;  + + i )  {   
			
		
	
		
		
			
				
					
					      std : : string  key  =  " key "  +  ToString ( i ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					      std : : string  blob_index ;   
			
		
	
		
		
			
				
					
					      BlobIndex : : EncodeInlinedTTL ( & blob_index ,  /* expiration */  9876543210 ,   
			
		
	
		
		
			
				
					
					                                  " blob "  +  ToString ( i ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					      key_values . emplace_back ( std : : move ( key ) ,  std : : move ( blob_index ) ) ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    for  ( const  auto &  key_value  :  key_values )  {   
			
		
	
		
		
			
				
					
					      WriteBatch  batch ;        WriteBatch  batch ;   
			
		
	
		
		
			
				
					
					      ASSERT_OK ( PutBlobIndex ( & batch ,  " key "  +  index ,  " blob "  +  index ) ) ;        ASSERT_OK ( PutBlobIndex ( & batch ,  key_value . first ,  key_value . second ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					      ASSERT_OK ( Write ( & batch ) ) ;        ASSERT_OK ( Write ( & batch ) ) ;   
			
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    MoveDataTo ( tier ) ;      MoveDataTo ( tier ) ;   
			
		
	
		
		
			
				
					
					    for  ( int  i  =  1 ;  i  < =  5 ;  i + + )  {  
  
			
				
				
			
		
	
		
		
			
				
					
					      std : : string  index  =  ToString ( i ) ;      for  ( const  auto &  key_value  :  key_values )  {   
			
				
				
			
		
	
		
		
			
				
					
					      ASSERT_EQ ( " blob "  +  index ,  GetBlobIndex ( " key "  +  index ) ) ;        ASSERT_EQ ( GetBlobIndex ( key_value . first ) ,  key_value . second ) ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					} }  
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -164,13 +184,19 @@ TEST_F(DBBlobIndexTest, Write) { 
			
		
	
		
		
			
				
					
					// accidentally opening the base DB of a stacked BlobDB and actual corruption
 // accidentally opening the base DB of a stacked BlobDB and actual corruption
  
			
		
	
		
		
			
				
					
					// when using the integrated BlobDB.
 // when using the integrated BlobDB.
  
			
		
	
		
		
			
				
					
					TEST_F ( DBBlobIndexTest ,  Get )  { TEST_F ( DBBlobIndexTest ,  Get )  {  
			
		
	
		
		
			
				
					
					  std : : string  blob_index ;   
			
		
	
		
		
			
				
					
					  BlobIndex : : EncodeInlinedTTL ( & blob_index ,  /* expiration */  9876543210 ,  " blob " ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  for  ( auto  tier  :  kAllTiers )  {    for  ( auto  tier  :  kAllTiers )  {   
			
		
	
		
		
			
				
					
					    DestroyAndReopen ( GetTestOptions ( ) ) ;      DestroyAndReopen ( GetTestOptions ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    WriteBatch  batch ;      WriteBatch  batch ;   
			
		
	
		
		
			
				
					
					    ASSERT_OK ( batch . Put ( " key " ,  " value " ) ) ;      ASSERT_OK ( batch . Put ( " key " ,  " value " ) ) ;   
			
		
	
		
		
			
				
					
					    ASSERT_OK ( PutBlobIndex ( & batch ,  " blob_key " ,  " blob_index " ) ) ;      ASSERT_OK ( PutBlobIndex ( & batch ,  " blob_key " ,  blob_index ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					    ASSERT_OK ( Write ( & batch ) ) ;      ASSERT_OK ( Write ( & batch ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    MoveDataTo ( tier ) ;      MoveDataTo ( tier ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    // Verify normal value
      // Verify normal value
   
			
		
	
		
		
			
				
					
					    bool  is_blob_index  =  false ;      bool  is_blob_index  =  false ;   
			
		
	
		
		
			
				
					
					    PinnableSlice  value ;      PinnableSlice  value ;   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -178,6 +204,7 @@ TEST_F(DBBlobIndexTest, Get) { 
			
		
	
		
		
			
				
					
					    ASSERT_EQ ( " value " ,  GetImpl ( " key " ) ) ;      ASSERT_EQ ( " value " ,  GetImpl ( " key " ) ) ;   
			
		
	
		
		
			
				
					
					    ASSERT_EQ ( " value " ,  GetImpl ( " key " ,  & is_blob_index ) ) ;      ASSERT_EQ ( " value " ,  GetImpl ( " key " ,  & is_blob_index ) ) ;   
			
		
	
		
		
			
				
					
					    ASSERT_FALSE ( is_blob_index ) ;      ASSERT_FALSE ( is_blob_index ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    // Verify blob index
      // Verify blob index
   
			
		
	
		
		
			
				
					
					    if  ( tier  < =  kImmutableMemtables )  {      if  ( tier  < =  kImmutableMemtables )  {   
			
		
	
		
		
			
				
					
					      ASSERT_TRUE ( Get ( " blob_key " ,  & value ) . IsNotSupported ( ) ) ;        ASSERT_TRUE ( Get ( " blob_key " ,  & value ) . IsNotSupported ( ) ) ;   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -186,7 +213,7 @@ TEST_F(DBBlobIndexTest, Get) { 
			
		
	
		
		
			
				
					
					      ASSERT_TRUE ( Get ( " blob_key " ,  & value ) . IsCorruption ( ) ) ;        ASSERT_TRUE ( Get ( " blob_key " ,  & value ) . IsCorruption ( ) ) ;   
			
		
	
		
		
			
				
					
					      ASSERT_EQ ( " CORRUPTION " ,  GetImpl ( " blob_key " ) ) ;        ASSERT_EQ ( " CORRUPTION " ,  GetImpl ( " blob_key " ) ) ;   
			
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					    ASSERT_EQ ( " blob_index " ,  GetImpl ( " blob_key " ,  & is_blob_index ) ) ;      ASSERT_EQ ( blob_index ,  GetImpl ( " blob_key " ,  & is_blob_index ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					    ASSERT_TRUE ( is_blob_index ) ;      ASSERT_TRUE ( is_blob_index ) ;   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					} }  
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -196,11 +223,14 @@ TEST_F(DBBlobIndexTest, Get) { 
			
		
	
		
		
			
				
					
					// if blob index is updated with a normal value. See the test case above for
 // if blob index is updated with a normal value. See the test case above for
  
			
		
	
		
		
			
				
					
					// more details.
 // more details.
  
			
		
	
		
		
			
				
					
					TEST_F ( DBBlobIndexTest ,  Updated )  { TEST_F ( DBBlobIndexTest ,  Updated )  {  
			
		
	
		
		
			
				
					
					  std : : string  blob_index ;   
			
		
	
		
		
			
				
					
					  BlobIndex : : EncodeInlinedTTL ( & blob_index ,  /* expiration */  9876543210 ,  " blob " ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  for  ( auto  tier  :  kAllTiers )  {    for  ( auto  tier  :  kAllTiers )  {   
			
		
	
		
		
			
				
					
					    DestroyAndReopen ( GetTestOptions ( ) ) ;      DestroyAndReopen ( GetTestOptions ( ) ) ;   
			
		
	
		
		
			
				
					
					    WriteBatch  batch ;      WriteBatch  batch ;   
			
		
	
		
		
			
				
					
					    for  ( int  i  =  0 ;  i  <  10 ;  i + + )  {      for  ( int  i  =  0 ;  i  <  10 ;  i + + )  {   
			
		
	
		
		
			
				
					
					      ASSERT_OK ( PutBlobIndex ( & batch ,  " key "  +  ToString ( i ) ,  " blob_index " ) ) ;        ASSERT_OK ( PutBlobIndex ( & batch ,  " key "  +  ToString ( i ) ,  blob_index ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					    ASSERT_OK ( Write ( & batch ) ) ;      ASSERT_OK ( Write ( & batch ) ) ;   
			
		
	
		
		
			
				
					
					    // Avoid blob values from being purged.
      // Avoid blob values from being purged.
   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -218,7 +248,7 @@ TEST_F(DBBlobIndexTest, Updated) { 
			
		
	
		
		
			
				
					
					    ASSERT_OK ( dbfull ( ) - > DeleteRange ( WriteOptions ( ) ,  cfh ( ) ,  " key6 " ,  " key9 " ) ) ;      ASSERT_OK ( dbfull ( ) - > DeleteRange ( WriteOptions ( ) ,  cfh ( ) ,  " key6 " ,  " key9 " ) ) ;   
			
		
	
		
		
			
				
					
					    MoveDataTo ( tier ) ;      MoveDataTo ( tier ) ;   
			
		
	
		
		
			
				
					
					    for  ( int  i  =  0 ;  i  <  10 ;  i + + )  {      for  ( int  i  =  0 ;  i  <  10 ;  i + + )  {   
			
		
	
		
		
			
				
					
					      ASSERT_EQ ( " blob_index " ,  GetBlobIndex ( " key "  +  ToString ( i ) ,  snapshot ) ) ;        ASSERT_EQ ( blob_index ,  GetBlobIndex ( " key "  +  ToString ( i ) ,  snapshot ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					    ASSERT_EQ ( " new_value " ,  Get ( " key1 " ) ) ;      ASSERT_EQ ( " new_value " ,  Get ( " key1 " ) ) ;   
			
		
	
		
		
			
				
					
					    if  ( tier  < =  kImmutableMemtables )  {      if  ( tier  < =  kImmutableMemtables )  {   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -232,7 +262,7 @@ TEST_F(DBBlobIndexTest, Updated) { 
			
		
	
		
		
			
				
					
					    for  ( int  i  =  6 ;  i  <  9 ;  i + + )  {      for  ( int  i  =  6 ;  i  <  9 ;  i + + )  {   
			
		
	
		
		
			
				
					
					      ASSERT_EQ ( " NOT_FOUND " ,  Get ( " key "  +  ToString ( i ) ) ) ;        ASSERT_EQ ( " NOT_FOUND " ,  Get ( " key "  +  ToString ( i ) ) ) ;   
			
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					    ASSERT_EQ ( " blob_index " ,  GetBlobIndex ( " key9 " ) ) ;      ASSERT_EQ ( blob_index ,  GetBlobIndex ( " key9 " ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					    dbfull ( ) - > ReleaseSnapshot ( snapshot ) ;      dbfull ( ) - > ReleaseSnapshot ( snapshot ) ;   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					} }