@ -48,9 +48,21 @@ static std::string GetDirName(const std::string filename) { 
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					Status  SyncDir ( const  std : : string &  dir )  { // Trim the tailing "/" in the end of `str`
  
			
				
				
			
		
	
		
		
			
				
					
					  // As this is a test it isn't required to *actually* sync this directory.
  static  std : : string  TrimDirname ( const  std : : string &  str )  {  
			
				
				
			
		
	
		
		
			
				
					
					  return  Status : : OK ( ) ;    size_t  found  =  str . find_last_not_of ( " / " ) ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					  if  ( found  = =  std : : string : : npos )  {   
			
		
	
		
		
			
				
					
					    return  str ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  return  str . substr ( 0 ,  found  +  1 ) ;   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					// Return pair <parent directory name, file name> of a full path.
  
			
		
	
		
		
			
				
					
					static  std : : pair < std : : string ,  std : : string >  GetDirAndName (  
			
		
	
		
		
			
				
					
					    const  std : : string &  name )  {   
			
		
	
		
		
			
				
					
					  std : : string  dirname  =  GetDirName ( name ) ;   
			
		
	
		
		
			
				
					
					  std : : string  fname  =  name . substr ( dirname . size ( )  +  1 ) ;   
			
		
	
		
		
			
				
					
					  return  std : : make_pair ( dirname ,  fname ) ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					// A basic file truncation function suitable for this test.
 // A basic file truncation function suitable for this test.
  
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -124,10 +136,22 @@ class TestWritableFile : public WritableFile { 
			
		
	
		
		
			
				
					
					  unique_ptr < WritableFile >  target_ ;    unique_ptr < WritableFile >  target_ ;   
			
		
	
		
		
			
				
					
					  bool  writable_file_opened_ ;    bool  writable_file_opened_ ;   
			
		
	
		
		
			
				
					
					  FaultInjectionTestEnv *  env_ ;    FaultInjectionTestEnv *  env_ ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  Status  SyncParent ( ) ;   
			
		
	
		
		
			
				
					
					} ; } ;  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					class  TestDirectory  :  public  Directory  {  
			
		
	
		
		
			
				
					
					 public :   
			
		
	
		
		
			
				
					
					  explicit  TestDirectory ( FaultInjectionTestEnv *  env ,  std : : string  dirname ,   
			
		
	
		
		
			
				
					
					                         Directory *  dir )   
			
		
	
		
		
			
				
					
					      :  env_ ( env ) ,  dirname_ ( dirname ) ,  dir_ ( dir )  { }   
			
		
	
		
		
			
				
					
					  ~ TestDirectory ( )  { }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  virtual  Status  Fsync ( )  override ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					 private :   
			
		
	
		
		
			
				
					
					  FaultInjectionTestEnv *  env_ ;   
			
		
	
		
		
			
				
					
					  std : : string  dirname_ ;   
			
		
	
		
		
			
				
					
					  unique_ptr < Directory >  dir_ ;   
			
		
	
		
		
			
				
					
					} ;  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					class  FaultInjectionTestEnv  :  public  EnvWrapper  { class  FaultInjectionTestEnv  :  public  EnvWrapper  {  
			
		
	
		
		
			
				
					
					 public :   public :   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -136,6 +160,18 @@ class FaultInjectionTestEnv : public EnvWrapper { 
			
		
	
		
		
			
				
					
					        filesystem_active_ ( true )  { }          filesystem_active_ ( true )  { }   
			
		
	
		
		
			
				
					
					  virtual  ~ FaultInjectionTestEnv ( )  {  }    virtual  ~ FaultInjectionTestEnv ( )  {  }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  Status  NewDirectory ( const  std : : string &  name ,   
			
		
	
		
		
			
				
					
					                      unique_ptr < Directory > *  result )  override  {   
			
		
	
		
		
			
				
					
					    unique_ptr < Directory >  r ;   
			
		
	
		
		
			
				
					
					    Status  s  =  target ( ) - > NewDirectory ( name ,  & r ) ;   
			
		
	
		
		
			
				
					
					    ASSERT_OK ( s ) ;   
			
		
	
		
		
			
				
					
					    if  ( ! s . ok ( ) )  {   
			
		
	
		
		
			
				
					
					      return  s ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					    result - > reset ( new  TestDirectory ( this ,  TrimDirname ( name ) ,  r . release ( ) ) ) ;   
			
		
	
		
		
			
				
					
					    return  Status : : OK ( ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  Status  NewWritableFile ( const  std : : string &  fname ,    Status  NewWritableFile ( const  std : : string &  fname ,   
			
		
	
		
		
			
				
					
					                         unique_ptr < WritableFile > *  result ,                           unique_ptr < WritableFile > *  result ,   
			
		
	
		
		
			
				
					
					                         const  EnvOptions &  soptions )  {                           const  EnvOptions &  soptions )  {   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -146,7 +182,10 @@ class FaultInjectionTestEnv : public EnvWrapper { 
			
		
	
		
		
			
				
					
					      // again then it will be truncated - so forget our saved state.
        // again then it will be truncated - so forget our saved state.
   
			
		
	
		
		
			
				
					
					      UntrackFile ( fname ) ;        UntrackFile ( fname ) ;   
			
		
	
		
		
			
				
					
					      MutexLock  l ( & mutex_ ) ;        MutexLock  l ( & mutex_ ) ;   
			
		
	
		
		
			
				
					
					      new_files_since_last_dir_sync_ . insert ( fname ) ;  
 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					      auto  dir_and_name  =  GetDirAndName ( fname ) ;   
			
		
	
		
		
			
				
					
					      auto &  list  =  dir_to_new_files_since_last_sync_ [ dir_and_name . first ] ;   
			
		
	
		
		
			
				
					
					      list . insert ( dir_and_name . second ) ;   
			
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					    return  s ;      return  s ;   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -170,10 +209,12 @@ class FaultInjectionTestEnv : public EnvWrapper { 
			
		
	
		
		
			
				
					
					        db_file_state_ . erase ( s ) ;          db_file_state_ . erase ( s ) ;   
			
		
	
		
		
			
				
					
					      }        }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					      if  ( new_files_since_last_dir_sync_ . erase ( s )  ! =  0 )  {        auto  sdn  =  GetDirAndName ( s ) ;   
			
				
				
			
		
	
		
		
			
				
					
					        assert ( new_files_since_last_dir_sync_ . find ( t )  = =        auto  tdn  =  GetDirAndName ( t ) ;   
			
				
				
			
		
	
		
		
			
				
					
					               new_files_since_last_dir_sync_ . end ( ) ) ;        if  ( dir_to_new_files_since_last_sync_ [ sdn . first ] . erase ( sdn . second )  ! =  0 )  {   
			
				
				
			
		
	
		
		
			
				
					
					        new_files_since_last_dir_sync_ . insert ( t ) ;          auto &  tlist  =  dir_to_new_files_since_last_sync_ [ tdn . first ] ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					        assert ( tlist . find ( tdn . second )  = =  tlist . end ( ) ) ;   
			
		
	
		
		
			
				
					
					        tlist . insert ( tdn . second ) ;   
			
		
	
		
		
			
				
					
					      }        }   
			
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					
 
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -201,40 +242,37 @@ class FaultInjectionTestEnv : public EnvWrapper { 
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  Status  DeleteFilesCreatedAfterLastDirSync ( )  {    Status  DeleteFilesCreatedAfterLastDirSync ( )  {   
			
		
	
		
		
			
				
					
					    // Because DeleteFile access this container make a copy to avoid deadlock
      // Because DeleteFile access this container make a copy to avoid deadlock
   
			
		
	
		
		
			
				
					
					    mutex_ . Lock ( ) ;      std : : map < std : : string ,  std : : set < std : : string > >  map_copy ;   
			
				
				
			
		
	
		
		
			
				
					
					    std : : set < std : : string >  new_files ( new_files_since_last_dir_sync_ . begin ( ) ,      {   
			
				
				
			
		
	
		
		
			
				
					
					                                    new_files_since_last_dir_sync_ . end ( ) ) ;        MutexLock  l ( & mutex_ ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    mutex_ . Unlock ( ) ;        map_copy . insert ( dir_to_new_files_since_last_sync_ . begin ( ) ,   
			
				
				
			
		
	
		
		
			
				
					
					    Status  s ;                        dir_to_new_files_since_last_sync_ . end ( ) ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    std : : set < std : : string > : : const_iterator  it ;   
			
		
	
		
		
			
				
					
					    for  ( it  =  new_files . begin ( ) ;  s . ok ( )  & &  it  ! =  new_files . end ( ) ;  + + it )  {   
			
		
	
		
		
			
				
					
					      s  =  DeleteFile ( * it ) ;   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					    return  s ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  void  DirWasSynced ( )  {      for  ( auto &  pair  :  map_copy )  {   
			
				
				
			
		
	
		
		
			
				
					
					    MutexLock  l ( & mutex_ ) ;        for  ( std : : string  name  :  pair . second )  {   
			
				
				
			
		
	
		
		
			
				
					
					    new_files_since_last_dir_sync_ . clear ( ) ;          Status  s  =  DeleteFile ( pair . first  +  " / "  +  name ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  }        }   
			
				
				
			
		
	
		
		
			
				
					
					
    }   
			
				
				
			
		
	
		
		
			
				
					
					  bool  IsFileCreatedSinceLastDirSync ( const  std : : string &  filename )  {      return  Status : : OK ( ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    MutexLock  l ( & mutex_ ) ;   
			
		
	
		
		
			
				
					
					    return  new_files_since_last_dir_sync_ . find ( filename )  ! =   
			
		
	
		
		
			
				
					
					           new_files_since_last_dir_sync_ . end ( ) ;   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  void  ResetState ( )  {    void  ResetState ( )  {   
			
		
	
		
		
			
				
					
					    MutexLock  l ( & mutex_ ) ;      MutexLock  l ( & mutex_ ) ;   
			
		
	
		
		
			
				
					
					    db_file_state_ . clear ( ) ;      db_file_state_ . clear ( ) ;   
			
		
	
		
		
			
				
					
					    new_files_since_last_dir _sync_ . clear ( ) ;      dir_to_ new_files_since_last_sync_. clear ( ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					    SetFilesystemActive ( true ) ;      SetFilesystemActive ( true ) ;   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  void  UntrackFile ( const  std : : string &  f )  {    void  UntrackFile ( const  std : : string &  f )  {   
			
		
	
		
		
			
				
					
					    MutexLock  l ( & mutex_ ) ;      MutexLock  l ( & mutex_ ) ;   
			
		
	
		
		
			
				
					
					    auto  dir_and_name  =  GetDirAndName ( f ) ;   
			
		
	
		
		
			
				
					
					    dir_to_new_files_since_last_sync_ [ dir_and_name . first ] . erase (   
			
		
	
		
		
			
				
					
					        dir_and_name . second ) ;   
			
		
	
		
		
			
				
					
					    db_file_state_ . erase ( f ) ;      db_file_state_ . erase ( f ) ;   
			
		
	
		
		
			
				
					
					    new_files_since_last_dir_sync_ . erase ( f ) ;    }   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  void  SyncDir ( const  std : : string &  dirname )  {   
			
		
	
		
		
			
				
					
					    dir_to_new_files_since_last_sync_ . erase ( dirname ) ;   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Setting the filesystem to inactive is the test equivalent to simulating a
    // Setting the filesystem to inactive is the test equivalent to simulating a
   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -247,7 +285,8 @@ class FaultInjectionTestEnv : public EnvWrapper { 
			
		
	
		
		
			
				
					
					 private :   private :   
			
		
	
		
		
			
				
					
					  port : : Mutex  mutex_ ;    port : : Mutex  mutex_ ;   
			
		
	
		
		
			
				
					
					  std : : map < std : : string ,  FileState >  db_file_state_ ;    std : : map < std : : string ,  FileState >  db_file_state_ ;   
			
		
	
		
		
			
				
					
					  std : : set < std : : string >  new_files_since_last_dir_sync_ ;    std : : unordered_map < std : : string ,  std : : set < std : : string > >   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					      dir_to_new_files_since_last_sync_ ;   
			
		
	
		
		
			
				
					
					  bool  filesystem_active_ ;   // Record flushes, syncs, writes
    bool  filesystem_active_ ;   // Record flushes, syncs, writes
   
			
		
	
		
		
			
				
					
					} ; } ;  
			
		
	
		
		
			
				
					
					
 
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -256,6 +295,11 @@ Status FileState::DropUnsyncedData() const { 
			
		
	
		
		
			
				
					
					  return  Truncate ( filename_ ,  sync_pos ) ;    return  Truncate ( filename_ ,  sync_pos ) ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					Status  TestDirectory : : Fsync ( )  {  
			
		
	
		
		
			
				
					
					  env_ - > SyncDir ( dirname_ ) ;   
			
		
	
		
		
			
				
					
					  return  dir_ - > Fsync ( ) ;   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TestWritableFile : : TestWritableFile ( const  std : : string &  fname , TestWritableFile : : TestWritableFile ( const  std : : string &  fname ,  
			
		
	
		
		
			
				
					
					                                   unique_ptr < WritableFile > & &  f ,                                     unique_ptr < WritableFile > & &  f ,   
			
		
	
		
		
			
				
					
					                                   FaultInjectionTestEnv *  env )                                     FaultInjectionTestEnv *  env )   
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -302,32 +346,35 @@ Status TestWritableFile::Sync() { 
			
		
	
		
		
			
				
					
					  if  ( ! env_ - > IsFilesystemActive ( ) )  {    if  ( ! env_ - > IsFilesystemActive ( ) )  {   
			
		
	
		
		
			
				
					
					    return  Status : : OK ( ) ;      return  Status : : OK ( ) ;   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					  // Ensure new files referred to by the manifest are in the filesystem.
    // No need to actual sync.
   
			
				
				
			
		
	
		
		
			
				
					
					  Status  s  =  target_ - > Sync ( ) ;    state_ . pos_at_last_sync_  =  state_ . pos_ ;   
			
				
				
			
		
	
		
		
			
				
					
					  if  ( s . ok ( ) )  {    return  Status : : OK ( ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    state_ . pos_at_last_sync_  =  state_ . pos_ ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  if  ( env_ - > IsFileCreatedSinceLastDirSync ( state_ . filename_ ) )  {   
			
		
	
		
		
			
				
					
					    Status  ps  =  SyncParent ( ) ;   
			
		
	
		
		
			
				
					
					    if  ( s . ok ( )  & &  ! ps . ok ( ) )  {   
			
		
	
		
		
			
				
					
					      s  =  ps ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  return  s ;   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					Status  TestWritableFile : : SyncParent ( )  {  
			
		
	
		
		
			
				
					
					  Status  s  =  SyncDir ( GetDirName ( state_ . filename_ ) ) ;   
			
		
	
		
		
			
				
					
					  if  ( s . ok ( ) )  {   
			
		
	
		
		
			
				
					
					    env_ - > DirWasSynced ( ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  return  s ;   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					class  FaultInjectionTest  { class  FaultInjectionTest  {  
			
		
	
		
		
			
				
					
					 protected :   
			
		
	
		
		
			
				
					
					  enum  OptionConfig  {   
			
		
	
		
		
			
				
					
					    kDefault ,   
			
		
	
		
		
			
				
					
					    kDifferentDataDir ,   
			
		
	
		
		
			
				
					
					    kWalDir ,   
			
		
	
		
		
			
				
					
					    kSyncWal ,   
			
		
	
		
		
			
				
					
					    kWalDirSyncWal ,   
			
		
	
		
		
			
				
					
					    kEnd ,   
			
		
	
		
		
			
				
					
					  } ;   
			
		
	
		
		
			
				
					
					  int  option_config_ ;   
			
		
	
		
		
			
				
					
					  // When need to make sure data is persistent, sync WAL
   
			
		
	
		
		
			
				
					
					  bool  sync_use_wal ;   
			
		
	
		
		
			
				
					
					  // When need to make sure data is persistent, call DB::CompactRange()
   
			
		
	
		
		
			
				
					
					  bool  sync_use_compact ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					 protected :   
			
		
	
		
		
			
				
					
					 public :   public :   
			
		
	
		
		
			
				
					
					  enum  ExpectedVerifResult  {  VAL_EXPECT_NO_ERROR ,  VAL_EXPECT_ERROR  } ;    enum  ExpectedVerifResult  {  kValExpectFound ,  kValExpectNoError  } ;   
			
				
				
			
		
	
		
		
			
				
					
					  enum  ResetMethod  {  RESET_DROP_UNSYNCED_DATA ,  RESET_DELETE_UNSYNCED_FILES  } ;    enum  ResetMethod  {   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					    kResetDropUnsyncedData ,   
			
		
	
		
		
			
				
					
					    kResetDeleteUnsyncedFiles ,   
			
		
	
		
		
			
				
					
					    kResetDropAndDeleteUnsynced   
			
		
	
		
		
			
				
					
					  } ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  FaultInjectionTestEnv *  env_ ;    FaultInjectionTestEnv *  env_ ;   
			
		
	
		
		
			
				
					
					  std : : string  dbname_ ;    std : : string  dbname_ ;   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -335,10 +382,54 @@ class FaultInjectionTest { 
			
		
	
		
		
			
				
					
					  Options  options_ ;    Options  options_ ;   
			
		
	
		
		
			
				
					
					  DB *  db_ ;    DB *  db_ ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  FaultInjectionTest ( )  :  env_ ( NULL ) ,  db_ ( NULL )  {  NewDB ( ) ;  }    FaultInjectionTest ( )   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					      :  option_config_ ( kDefault ) ,   
			
		
	
		
		
			
				
					
					        sync_use_wal ( false ) ,   
			
		
	
		
		
			
				
					
					        sync_use_compact ( true ) ,   
			
		
	
		
		
			
				
					
					        env_ ( NULL ) ,   
			
		
	
		
		
			
				
					
					        db_ ( NULL )  {   
			
		
	
		
		
			
				
					
					    NewDB ( ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ~ FaultInjectionTest ( )  {  ASSERT_OK ( TearDown ( ) ) ;  }    ~ FaultInjectionTest ( )  {  ASSERT_OK ( TearDown ( ) ) ;  }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  bool  ChangeOptions ( )  {   
			
		
	
		
		
			
				
					
					    option_config_ + + ;   
			
		
	
		
		
			
				
					
					    if  ( option_config_  > =  kEnd )  {   
			
		
	
		
		
			
				
					
					      return  false ;   
			
		
	
		
		
			
				
					
					    }  else  {   
			
		
	
		
		
			
				
					
					      return  true ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Return the current option configuration.
   
			
		
	
		
		
			
				
					
					  Options  CurrentOptions ( )  {   
			
		
	
		
		
			
				
					
					    sync_use_wal  =  false ;   
			
		
	
		
		
			
				
					
					    sync_use_compact  =  true ;   
			
		
	
		
		
			
				
					
					    Options  options ;   
			
		
	
		
		
			
				
					
					    switch  ( option_config_ )  {   
			
		
	
		
		
			
				
					
					      case  kWalDir :   
			
		
	
		
		
			
				
					
					        options . wal_dir  =  test : : TmpDir ( env_ )  +  " /fault_test_wal " ;   
			
		
	
		
		
			
				
					
					        break ;   
			
		
	
		
		
			
				
					
					      case  kDifferentDataDir :   
			
		
	
		
		
			
				
					
					        options . db_paths . emplace_back ( test : : TmpDir ( env_ )  +  " /fault_test_data " ,   
			
		
	
		
		
			
				
					
					                                      1000000U ) ;   
			
		
	
		
		
			
				
					
					        break ;   
			
		
	
		
		
			
				
					
					      case  kSyncWal :   
			
		
	
		
		
			
				
					
					        sync_use_wal  =  true ;   
			
		
	
		
		
			
				
					
					        sync_use_compact  =  false ;   
			
		
	
		
		
			
				
					
					        break ;   
			
		
	
		
		
			
				
					
					      case  kWalDirSyncWal :   
			
		
	
		
		
			
				
					
					        options . wal_dir  =  test : : TmpDir ( env_ )  +  " /fault_test_wal " ;   
			
		
	
		
		
			
				
					
					        sync_use_wal  =  true ;   
			
		
	
		
		
			
				
					
					        sync_use_compact  =  false ;   
			
		
	
		
		
			
				
					
					        break ;   
			
		
	
		
		
			
				
					
					      default :   
			
		
	
		
		
			
				
					
					        break ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					    return  options ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  Status  NewDB ( )  {    Status  NewDB ( )  {   
			
		
	
		
		
			
				
					
					    assert ( db_  = =  NULL ) ;      assert ( db_  = =  NULL ) ;   
			
		
	
		
		
			
				
					
					    assert ( tiny_cache_  = =  nullptr ) ;      assert ( tiny_cache_  = =  nullptr ) ;   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -346,7 +437,7 @@ class FaultInjectionTest { 
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    env_  =  new  FaultInjectionTestEnv ( Env : : Default ( ) ) ;      env_  =  new  FaultInjectionTestEnv ( Env : : Default ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    options_  =  Options ( ) ;      options_  =  Current Options( ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					    options_ . env  =  env_ ;      options_ . env  =  env_ ;   
			
		
	
		
		
			
				
					
					    options_ . paranoid_checks  =  true ;      options_ . paranoid_checks  =  true ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -357,6 +448,8 @@ class FaultInjectionTest { 
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    dbname_  =  test : : TmpDir ( )  +  " /fault_test " ;      dbname_  =  test : : TmpDir ( )  +  " /fault_test " ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    ASSERT_OK ( DestroyDB ( dbname_ ,  options_ ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    options_ . create_if_missing  =  true ;      options_ . create_if_missing  =  true ;   
			
		
	
		
		
			
				
					
					    Status  s  =  OpenDB ( ) ;      Status  s  =  OpenDB ( ) ;   
			
		
	
		
		
			
				
					
					    options_ . create_if_missing  =  false ;      options_ . create_if_missing  =  false ;   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -374,7 +467,7 @@ class FaultInjectionTest { 
			
		
	
		
		
			
				
					
					  Status  TearDown ( )  {    Status  TearDown ( )  {   
			
		
	
		
		
			
				
					
					    CloseDB ( ) ;      CloseDB ( ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    Status  s  =  DestroyDB ( dbname_ ,  Options ( ) ) ;      Status  s  =  DestroyDB ( dbname_ ,  options_ ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    delete  env_ ;      delete  env_ ;   
			
		
	
		
		
			
				
					
					    env_  =  NULL ;      env_  =  NULL ;   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -384,15 +477,14 @@ class FaultInjectionTest { 
			
		
	
		
		
			
				
					
					    return  s ;      return  s ;   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  void  Build ( int  start_idx ,  int  num_vals )  {    void  Build ( const  WriteOptions &  write_options ,  int  start_idx ,  int  num_vals )  {   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					    std : : string  key_space ,  value_space ;      std : : string  key_space ,  value_space ;   
			
		
	
		
		
			
				
					
					    WriteBatch  batch ;      WriteBatch  batch ;   
			
		
	
		
		
			
				
					
					    for  ( int  i  =  start_idx ;  i  <  start_idx  +  num_vals ;  i + + )  {      for  ( int  i  =  start_idx ;  i  <  start_idx  +  num_vals ;  i + + )  {   
			
		
	
		
		
			
				
					
					      Slice  key  =  Key ( i ,  & key_space ) ;        Slice  key  =  Key ( i ,  & key_space ) ;   
			
		
	
		
		
			
				
					
					      batch . Clear ( ) ;        batch . Clear ( ) ;   
			
		
	
		
		
			
				
					
					      batch . Put ( key ,  Value ( i ,  & value_space ) ) ;        batch . Put ( key ,  Value ( i ,  & value_space ) ) ;   
			
		
	
		
		
			
				
					
					      WriteOptions  options ;        ASSERT_OK ( db_ - > Write ( write_options ,  & batch ) ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      ASSERT_OK ( db_ - > Write ( options ,  & batch ) ) ;   
			
		
	
		
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -412,18 +504,22 @@ class FaultInjectionTest { 
			
		
	
		
		
			
				
					
					    for  ( int  i  =  start_idx ;  i  <  start_idx  +  num_vals  & &  s . ok ( ) ;  i + + )  {      for  ( int  i  =  start_idx ;  i  <  start_idx  +  num_vals  & &  s . ok ( ) ;  i + + )  {   
			
		
	
		
		
			
				
					
					      Value ( i ,  & value_space ) ;        Value ( i ,  & value_space ) ;   
			
		
	
		
		
			
				
					
					      s  =  ReadValue ( i ,  & val ) ;        s  =  ReadValue ( i ,  & val ) ;   
			
		
	
		
		
			
				
					
					      if  ( expected  = =  VAL_EXPECT_NO_ERROR )  {        if  ( s . ok ( ) )  {   
			
				
				
			
		
	
		
		
			
				
					
					        if  ( s . ok ( ) )  {          ASSERT_EQ ( value_space ,  val ) ;   
			
				
				
			
		
	
		
		
			
				
					
					          ASSERT_EQ ( value_space ,  val ) ;        }   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					      if  ( expected  = =  kValExpectFound )  {   
			
		
	
		
		
			
				
					
					        if  ( ! s . ok ( ) )  {   
			
		
	
		
		
			
				
					
					          fprintf ( stderr ,  " Error when read %dth record (expect found): %s \n " ,  i ,   
			
		
	
		
		
			
				
					
					                  s . ToString ( ) . c_str ( ) ) ;   
			
		
	
		
		
			
				
					
					          return  s ;   
			
		
	
		
		
			
				
					
					        }          }   
			
		
	
		
		
			
				
					
					      }  else  if  ( s . ok ( ) )  {        }  else  if  ( ! s . ok ( )  & &  ! s . IsNotFound ( ) )  {   
			
				
				
			
		
	
		
		
			
				
					
					        fprintf ( stderr ,  " Expected an error at %d, but was OK \n " ,  i ) ;          fprintf ( stderr ,  " Error when read %dth record: %s \n " ,  i ,   
			
				
				
			
		
	
		
		
			
				
					
					        s  =  Status : : IOError ( dbname_ ,  " Expected value error: " ) ;                  s . ToString ( ) . c_str ( ) ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      }  else  {          return  s ;   
			
				
				
			
		
	
		
		
			
				
					
					        s  =  Status : : OK ( ) ;   // An expected error
   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					      }        }   
			
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					    return  s ;      return  Statu s: : OK ( ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Return the ith key
    // Return the ith key
   
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -460,14 +556,22 @@ class FaultInjectionTest { 
			
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    delete  iter ;      delete  iter ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    FlushOptions  flush_options ;   
			
		
	
		
		
			
				
					
					    flush_options . wait  =  true ;   
			
		
	
		
		
			
				
					
					    db_ - > Flush ( flush_options ) ;   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  void  ResetDBState ( ResetMethod  reset_method )  {    void  ResetDBState ( ResetMethod  reset_method )  {   
			
		
	
		
		
			
				
					
					    switch  ( reset_method )  {      switch  ( reset_method )  {   
			
		
	
		
		
			
				
					
					      case  RESET_DROP_UNSYNCED_DATA :        case  kResetDropUnsyncedData :   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					        ASSERT_OK ( env_ - > DropUnsyncedFileData ( ) ) ;          ASSERT_OK ( env_ - > DropUnsyncedFileData ( ) ) ;   
			
		
	
		
		
			
				
					
					        break ;          break ;   
			
		
	
		
		
			
				
					
					      case  RESET_DELETE_UNSYNCED_FILES :        case  kResetDeleteUnsyncedFiles :   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					        ASSERT_OK ( env_ - > DeleteFilesCreatedAfterLastDirSync ( ) ) ;   
			
		
	
		
		
			
				
					
					        break ;   
			
		
	
		
		
			
				
					
					      case  kResetDropAndDeleteUnsynced :   
			
		
	
		
		
			
				
					
					        ASSERT_OK ( env_ - > DropUnsyncedFileData ( ) ) ;   
			
		
	
		
		
			
				
					
					        ASSERT_OK ( env_ - > DeleteFilesCreatedAfterLastDirSync ( ) ) ;          ASSERT_OK ( env_ - > DeleteFilesCreatedAfterLastDirSync ( ) ) ;   
			
		
	
		
		
			
				
					
					        break ;          break ;   
			
		
	
		
		
			
				
					
					      default :        default :   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -477,9 +581,16 @@ class FaultInjectionTest { 
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  void  PartialCompactTestPreFault ( int  num_pre_sync ,  int  num_post_sync )  {    void  PartialCompactTestPreFault ( int  num_pre_sync ,  int  num_post_sync )  {   
			
		
	
		
		
			
				
					
					    DeleteAllData ( ) ;      DeleteAllData ( ) ;   
			
		
	
		
		
			
				
					
					    Build ( 0 ,  num_pre_sync ) ;  
 
			
				
				
			
		
	
		
		
			
				
					
					    db_ - > CompactRange ( NULL ,  NULL ) ;      WriteOptions  write_options ;   
			
				
				
			
		
	
		
		
			
				
					
					    Build ( num_pre_sync ,  num_post_sync ) ;      write_options . sync  =  sync_use_wal ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    Build ( write_options ,  0 ,  num_pre_sync ) ;   
			
		
	
		
		
			
				
					
					    if  ( sync_use_compact )  {   
			
		
	
		
		
			
				
					
					      db_ - > CompactRange ( nullptr ,  nullptr ) ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					    write_options . sync  =  false ;   
			
		
	
		
		
			
				
					
					    Build ( write_options ,  num_pre_sync ,  num_post_sync ) ;   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  void  PartialCompactTestReopenWithFault ( ResetMethod  reset_method ,    void  PartialCompactTestReopenWithFault ( ResetMethod  reset_method ,   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -489,9 +600,9 @@ class FaultInjectionTest { 
			
		
	
		
		
			
				
					
					    CloseDB ( ) ;      CloseDB ( ) ;   
			
		
	
		
		
			
				
					
					    ResetDBState ( reset_method ) ;      ResetDBState ( reset_method ) ;   
			
		
	
		
		
			
				
					
					    ASSERT_OK ( OpenDB ( ) ) ;      ASSERT_OK ( OpenDB ( ) ) ;   
			
		
	
		
		
			
				
					
					    ASSERT_OK ( Verify ( 0 ,  num_pre_sync ,  FaultInjectionTest : : VAL_EXPECT_NO_ERROR ) ) ;      ASSERT_OK ( Verify ( 0 ,  num_pre_sync ,  FaultInjectionTest : : kValExpectFound ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					    ASSERT_OK ( Verify ( num_pre_sync ,  num_post_sync ,      ASSERT_OK ( Verify ( num_pre_sync ,  num_post_sync ,   
			
		
	
		
		
			
				
					
					      FaultInjectionTest : : VAL_EXPECT_ERROR ) ) ;                        FaultInjectionTest : : kValExpectNoError ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  void  NoWriteTestPreFault ( )  {    void  NoWriteTestPreFault ( )  {   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -505,30 +616,45 @@ class FaultInjectionTest { 
			
		
	
		
		
			
				
					
					} ; } ;  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST ( FaultInjectionTest ,  FaultTest )  { TEST ( FaultInjectionTest ,  FaultTest )  {  
			
		
	
		
		
			
				
					
					  Random  rnd ( 0 ) ;    do  {   
			
				
				
			
		
	
		
		
			
				
					
					  ASSERT_OK ( SetUp ( ) ) ;      Random  rnd ( 301 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  for  ( size_t  idx  =  0 ;  idx  <  kNumIterations ;  idx + + )  {      ASSERT_OK ( SetUp ( ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    int  num_pre_sync  =  rnd . Uniform ( kMaxNumValues ) ;      int  num_pre_sync  =  rnd . Uniform ( kMaxNumValues ) ;   
			
		
	
		
		
			
				
					
					    int  num_post_sync  =  rnd . Uniform ( kMaxNumValues ) ;      int  num_post_sync  =  rnd . Uniform ( kMaxNumValues ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    PartialCompactTestPreFault ( num_pre_sync ,  num_post_sync ) ;      PartialCompactTestPreFault ( num_pre_sync ,  num_post_sync ) ;   
			
		
	
		
		
			
				
					
					    PartialCompactTestReopenWithFault ( RESET_DROP_UNSYNCED_DATA ,      PartialCompactTestReopenWithFault ( kResetDropUnsyncedData ,  num_pre_sync ,   
			
				
				
			
		
	
		
		
			
				
					
					                                      num_pre_sync ,   
			
		
	
		
		
	
		
		
			
				
					
					                                      num_post_sync ) ;                                        num_post_sync ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    NoWriteTestPreFault ( ) ;   
			
		
	
		
		
			
				
					
					    NoWriteTestReopenWithFault ( RESET_DROP_UNSYNCED_DATA ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    PartialCompactTestPreFault ( num_pre_sync ,  num_post_sync ) ;   
			
		
	
		
		
			
				
					
					    // No new files created so we expect all values since no files will be
   
			
		
	
		
		
			
				
					
					    // dropped.
   
			
		
	
		
		
			
				
					
					    PartialCompactTestReopenWithFault ( RESET_DELETE_UNSYNCED_FILES ,   
			
		
	
		
		
			
				
					
					                                      num_pre_sync  +  num_post_sync ,   
			
		
	
		
		
			
				
					
					                                      0 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    NoWriteTestPreFault ( ) ;      NoWriteTestPreFault ( ) ;   
			
		
	
		
		
			
				
					
					    NoWriteTestReopenWithFault ( RESET_DELETE_UNSYNCED_FILES ) ;      NoWriteTestReopenWithFault ( kResetDropUnsyncedData ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  }  
 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					    // TODO(t6070540) Need to sync WAL Dir and other DB paths too.
   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					    // Setting a separate data path won't pass the test as we don't sync
   
			
		
	
		
		
			
				
					
					    // it after creating new files,
   
			
		
	
		
		
			
				
					
					    if  ( option_config_  ! =  kDifferentDataDir )  {   
			
		
	
		
		
			
				
					
					      PartialCompactTestPreFault ( num_pre_sync ,  num_post_sync ) ;   
			
		
	
		
		
			
				
					
					      // Since we don't sync WAL Dir, this test dosn't pass.
   
			
		
	
		
		
			
				
					
					      if  ( option_config_  ! =  kWalDirSyncWal )  {   
			
		
	
		
		
			
				
					
					        PartialCompactTestReopenWithFault ( kResetDropAndDeleteUnsynced ,   
			
		
	
		
		
			
				
					
					                                          num_pre_sync ,  num_post_sync ) ;   
			
		
	
		
		
			
				
					
					      }   
			
		
	
		
		
			
				
					
					      NoWriteTestPreFault ( ) ;   
			
		
	
		
		
			
				
					
					      NoWriteTestReopenWithFault ( kResetDropAndDeleteUnsynced ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					      PartialCompactTestPreFault ( num_pre_sync ,  num_post_sync ) ;   
			
		
	
		
		
			
				
					
					      // No new files created so we expect all values since no files will be
   
			
		
	
		
		
			
				
					
					      // dropped.
   
			
		
	
		
		
			
				
					
					      // WAL Dir is not synced for now.
   
			
		
	
		
		
			
				
					
					      if  ( option_config_  ! =  kWalDir  & &  option_config_  ! =  kWalDirSyncWal )  {   
			
		
	
		
		
			
				
					
					        PartialCompactTestReopenWithFault ( kResetDeleteUnsyncedFiles ,   
			
		
	
		
		
			
				
					
					                                          num_pre_sync  +  num_post_sync ,  0 ) ;   
			
		
	
		
		
			
				
					
					      }   
			
		
	
		
		
			
				
					
					      NoWriteTestPreFault ( ) ;   
			
		
	
		
		
			
				
					
					      NoWriteTestReopenWithFault ( kResetDeleteUnsyncedFiles ) ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					  }  while  ( ChangeOptions ( ) ) ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					}   // namespace rocksdb
 }   // namespace rocksdb