@ -1195,6 +1195,175 @@ TEST(DBTest, RepeatedWritesToSameKey) { 
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					// This is a static filter used for filtering
  
			
		
	
		
		
			
				
					
					// kvs during the compaction process.
  
			
		
	
		
		
			
				
					
					static  int  cfilter_count ;  
			
		
	
		
		
			
				
					
					static  std : : string  NEW_VALUE  =  " NewValue " ;  
			
		
	
		
		
			
				
					
					static  bool  keep_filter ( int  level ,  const  Slice &  key ,  
			
		
	
		
		
			
				
					
					  const  Slice &  value ,  Slice * *  new_value )  {   
			
		
	
		
		
			
				
					
					  cfilter_count + + ;   
  
			
		
	
		
		
			
				
					
					  return  false ;   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					static  bool  delete_filter ( int  level ,  const  Slice &  key ,  
			
		
	
		
		
			
				
					
					  const  Slice &  value ,  Slice * *  new_value )  {   
			
		
	
		
		
			
				
					
					  cfilter_count + + ;   
  
			
		
	
		
		
			
				
					
					  return  true ;   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					static  bool  change_filter ( int  level ,  const  Slice &  key ,  
			
		
	
		
		
			
				
					
					  const  Slice &  value ,  Slice * *  new_value )  {   
			
		
	
		
		
			
				
					
					  assert ( new_value  ! =  NULL ) ;   
			
		
	
		
		
			
				
					
					  * new_value  =  new  Slice ( NEW_VALUE ) ;   
			
		
	
		
		
			
				
					
					  return  false ;   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST ( DBTest ,  CompactionFilter )  {  
			
		
	
		
		
			
				
					
					  Options  options  =  CurrentOptions ( ) ;   
			
		
	
		
		
			
				
					
					  options . num_levels  =  3 ;   
			
		
	
		
		
			
				
					
					  options . max_mem_compaction_level  =  0 ;   
			
		
	
		
		
			
				
					
					  options . CompactionFilter  =  keep_filter ;   
			
		
	
		
		
			
				
					
					  Reopen ( & options ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Write 100K+1 keys, these are written to a few files 
   
			
		
	
		
		
			
				
					
					  // in L0. We do this so that the current snapshot points 
   
			
		
	
		
		
			
				
					
					  // to the 100001 key.The compaction filter is  not invoked
   
			
		
	
		
		
			
				
					
					  // on keys that are visible via a snapshot because we
   
			
		
	
		
		
			
				
					
					  // anyways cannot delete it.
   
			
		
	
		
		
			
				
					
					  const  std : : string  value ( 10 ,  ' x ' ) ;   
			
		
	
		
		
			
				
					
					  for  ( int  i  =  0 ;  i  <  100001 ;  i + + )  {   
			
		
	
		
		
			
				
					
					    char  key [ 100 ] ;   
			
		
	
		
		
			
				
					
					    snprintf ( key ,  sizeof ( key ) ,  " B%010d " ,  i ) ;   
			
		
	
		
		
			
				
					
					    Put ( key ,  value ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactMemTable ( ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Push all files to the highest level L2. Verify that
   
			
		
	
		
		
			
				
					
					  // the compaction is each level invokes the filter for
   
			
		
	
		
		
			
				
					
					  // all the keys in that level.
   
			
		
	
		
		
			
				
					
					  cfilter_count  =  0 ;   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactRange ( 0 ,  NULL ,  NULL ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( cfilter_count ,  100000 ) ;   
			
		
	
		
		
			
				
					
					  cfilter_count  =  0 ;   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactRange ( 1 ,  NULL ,  NULL ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( cfilter_count ,  100000 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( NumTableFilesAtLevel ( 0 ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( NumTableFilesAtLevel ( 1 ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_NE ( NumTableFilesAtLevel ( 2 ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					  cfilter_count  =  0 ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // overwrite all the 100K+1 keys once again.
   
			
		
	
		
		
			
				
					
					  for  ( int  i  =  0 ;  i  <  100001 ;  i + + )  {   
			
		
	
		
		
			
				
					
					    char  key [ 100 ] ;   
			
		
	
		
		
			
				
					
					    snprintf ( key ,  sizeof ( key ) ,  " B%010d " ,  i ) ;   
			
		
	
		
		
			
				
					
					    Put ( key ,  value ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactMemTable ( ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // push all files to the highest level L2. This
   
			
		
	
		
		
			
				
					
					  // means that all keys should pass at least once
   
			
		
	
		
		
			
				
					
					  // via the compaction filter
   
			
		
	
		
		
			
				
					
					  cfilter_count  =  0 ;   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactRange ( 0 ,  NULL ,  NULL ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( cfilter_count ,  100000 ) ;   
			
		
	
		
		
			
				
					
					  cfilter_count  =  0 ;   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactRange ( 1 ,  NULL ,  NULL ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( cfilter_count ,  100000 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( NumTableFilesAtLevel ( 0 ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( NumTableFilesAtLevel ( 1 ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_NE ( NumTableFilesAtLevel ( 2 ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // create a new database with the compaction
   
			
		
	
		
		
			
				
					
					  // filter in such a way that it deletes all keys
   
			
		
	
		
		
			
				
					
					  options . CompactionFilter  =  delete_filter ;   
			
		
	
		
		
			
				
					
					  options . create_if_missing  =  true ;   
			
		
	
		
		
			
				
					
					  DestroyAndReopen ( & options ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // write all the keys once again.
   
			
		
	
		
		
			
				
					
					  for  ( int  i  =  0 ;  i  <  100001 ;  i + + )  {   
			
		
	
		
		
			
				
					
					    char  key [ 100 ] ;   
			
		
	
		
		
			
				
					
					    snprintf ( key ,  sizeof ( key ) ,  " B%010d " ,  i ) ;   
			
		
	
		
		
			
				
					
					    Put ( key ,  value ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactMemTable ( ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_NE ( NumTableFilesAtLevel ( 0 ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( NumTableFilesAtLevel ( 1 ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( NumTableFilesAtLevel ( 2 ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Push all files to the highest level L2. This
   
			
		
	
		
		
			
				
					
					  // triggers the compaction filter to delete all keys,
   
			
		
	
		
		
			
				
					
					  // verify that at the end of the compaction process,
   
			
		
	
		
		
			
				
					
					  // nothing is left.
   
			
		
	
		
		
			
				
					
					  cfilter_count  =  0 ;   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactRange ( 0 ,  NULL ,  NULL ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( cfilter_count ,  100000 ) ;   
			
		
	
		
		
			
				
					
					  cfilter_count  =  0 ;   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactRange ( 1 ,  NULL ,  NULL ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( cfilter_count ,  0 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( NumTableFilesAtLevel ( 0 ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( NumTableFilesAtLevel ( 1 ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Scan the entire database to ensure that only the
   
			
		
	
		
		
			
				
					
					  // 100001th key is left in the db. The 100001th key
   
			
		
	
		
		
			
				
					
					  // is part of the default-most-current snapshot and
   
			
		
	
		
		
			
				
					
					  // cannot be deleted.
   
			
		
	
		
		
			
				
					
					  Iterator *  iter  =  db_ - > NewIterator ( ReadOptions ( ) ) ;   
			
		
	
		
		
			
				
					
					  iter - > SeekToFirst ( ) ;   
			
		
	
		
		
			
				
					
					  int  count  =  0 ;   
			
		
	
		
		
			
				
					
					  while  ( iter - > Valid ( ) )  {   
			
		
	
		
		
			
				
					
					    count + + ;   
			
		
	
		
		
			
				
					
					    iter - > Next ( ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( count ,  1 ) ;   
			
		
	
		
		
			
				
					
					  delete  iter ;   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST ( DBTest ,  CompactionFilterWithValueChange )  {  
			
		
	
		
		
			
				
					
					  Options  options  =  CurrentOptions ( ) ;   
			
		
	
		
		
			
				
					
					  options . num_levels  =  3 ;   
			
		
	
		
		
			
				
					
					  options . max_mem_compaction_level  =  0 ;   
			
		
	
		
		
			
				
					
					  options . CompactionFilter  =  change_filter ;   
			
		
	
		
		
			
				
					
					  Reopen ( & options ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Write 100K+1 keys, these are written to a few files 
   
			
		
	
		
		
			
				
					
					  // in L0. We do this so that the current snapshot points 
   
			
		
	
		
		
			
				
					
					  // to the 100001 key.The compaction filter is  not invoked
   
			
		
	
		
		
			
				
					
					  // on keys that are visible via a snapshot because we
   
			
		
	
		
		
			
				
					
					  // anyways cannot delete it.
   
			
		
	
		
		
			
				
					
					  const  std : : string  value ( 10 ,  ' x ' ) ;   
			
		
	
		
		
			
				
					
					  for  ( int  i  =  0 ;  i  <  100001 ;  i + + )  {   
			
		
	
		
		
			
				
					
					    char  key [ 100 ] ;   
			
		
	
		
		
			
				
					
					    snprintf ( key ,  sizeof ( key ) ,  " B%010d " ,  i ) ;   
			
		
	
		
		
			
				
					
					    Put ( key ,  value ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // push all files to  lower levels
   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactMemTable ( ) ;   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactRange ( 0 ,  NULL ,  NULL ) ;   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactRange ( 1 ,  NULL ,  NULL ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // re-write all data again
   
			
		
	
		
		
			
				
					
					  for  ( int  i  =  0 ;  i  <  100001 ;  i + + )  {   
			
		
	
		
		
			
				
					
					    char  key [ 100 ] ;   
			
		
	
		
		
			
				
					
					    snprintf ( key ,  sizeof ( key ) ,  " B%010d " ,  i ) ;   
			
		
	
		
		
			
				
					
					    Put ( key ,  value ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // push all files to  lower levels. This should
   
			
		
	
		
		
			
				
					
					  // invoke the compaction filter for all 100000 keys.
   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactMemTable ( ) ;   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactRange ( 0 ,  NULL ,  NULL ) ;   
			
		
	
		
		
			
				
					
					  dbfull ( ) - > TEST_CompactRange ( 1 ,  NULL ,  NULL ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // verify that all keys now have the new value that
   
			
		
	
		
		
			
				
					
					  // was set by the compaction process.
   
			
		
	
		
		
			
				
					
					  for  ( int  i  =  0 ;  i  <  100000 ;  i + + )  {   
			
		
	
		
		
			
				
					
					    char  key [ 100 ] ;   
			
		
	
		
		
			
				
					
					    snprintf ( key ,  sizeof ( key ) ,  " B%010d " ,  i ) ;   
			
		
	
		
		
			
				
					
					    std : : string  newvalue  =  Get ( key ) ;   
			
		
	
		
		
			
				
					
					    ASSERT_EQ ( newvalue . compare ( NEW_VALUE ) ,  0 ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST ( DBTest ,  SparseMerge )  { TEST ( DBTest ,  SparseMerge )  {  
			
		
	
		
		
			
				
					
					  Options  options  =  CurrentOptions ( ) ;    Options  options  =  CurrentOptions ( ) ;   
			
		
	
		
		
			
				
					
					  options . compression  =  kNoCompression ;    options . compression  =  kNoCompression ;