@ -21,6 +21,7 @@ TransactionLogIteratorImpl::TransactionLogIteratorImpl(
files_ ( std : : move ( files ) ) ,
files_ ( std : : move ( files ) ) ,
started_ ( false ) ,
started_ ( false ) ,
isValid_ ( false ) ,
isValid_ ( false ) ,
is_obsolete_ ( false ) ,
currentFileIndex_ ( 0 ) ,
currentFileIndex_ ( 0 ) ,
currentBatchSeq_ ( 0 ) ,
currentBatchSeq_ ( 0 ) ,
currentLastSeq_ ( 0 ) ,
currentLastSeq_ ( 0 ) ,
@ -69,14 +70,15 @@ bool TransactionLogIteratorImpl::Valid() {
return started_ & & isValid_ ;
return started_ & & isValid_ ;
}
}
bool TransactionLogIteratorImpl : : RestrictedRead (
bool TransactionLogIteratorImpl : : RestrictedRead ( Slice * record ,
Slice * record ,
std : : string * scratch ) {
std : : string * scratch ) {
// Don't read if no more complete entries to read from logs
bool ret = currentLogReader_ - > ReadRecord ( record , scratch ) ;
if ( currentLastSeq_ > = dbimpl_ - > GetLatestSequenceNumber ( ) ) {
return false ;
if ( ! reporter_ . last_status . ok ( ) ) {
currentStatus_ = reporter_ . last_status ;
}
}
return currentLogReader_ - > ReadRecord ( record , scratch ) ;
return ret ;
}
}
void TransactionLogIteratorImpl : : SeekToStartSequence (
void TransactionLogIteratorImpl : : SeekToStartSequence (
@ -86,6 +88,7 @@ void TransactionLogIteratorImpl::SeekToStartSequence(
Slice record ;
Slice record ;
started_ = false ;
started_ = false ;
isValid_ = false ;
isValid_ = false ;
is_obsolete_ = false ;
if ( files_ - > size ( ) < = startFileIndex ) {
if ( files_ - > size ( ) < = startFileIndex ) {
return ;
return ;
}
}
@ -94,6 +97,18 @@ void TransactionLogIteratorImpl::SeekToStartSequence(
currentStatus_ = s ;
currentStatus_ = s ;
return ;
return ;
}
}
auto latest_seq_num = dbimpl_ - > GetLatestSequenceNumber ( ) ;
if ( startingSequenceNumber_ > latest_seq_num ) {
if ( strict ) {
currentStatus_ = Status : : Corruption ( " Gap in sequence number. Could not "
" seek to required sequence number " ) ;
reporter_ . Info ( currentStatus_ . ToString ( ) . c_str ( ) ) ;
} else {
// isValid_ is false;
return ;
}
}
while ( RestrictedRead ( & record , & scratch ) ) {
while ( RestrictedRead ( & record , & scratch ) ) {
if ( record . size ( ) < 12 ) {
if ( record . size ( ) < 12 ) {
reporter_ . Corruption (
reporter_ . Corruption (
@ -123,11 +138,11 @@ void TransactionLogIteratorImpl::SeekToStartSequence(
// only file. Otherwise log the error and let the iterator return next entry
// only file. Otherwise log the error and let the iterator return next entry
// If strict is set, we want to seek exactly till the start sequence and it
// If strict is set, we want to seek exactly till the start sequence and it
// should have been present in the file we scanned above
// should have been present in the file we scanned above
if ( strict ) {
if ( strict | | files_ - > size ( ) = = 1 ) {
currentStatus_ = Status : : Corruption ( " Gap in sequence number. Could not "
currentStatus_ = Status : : Corruption ( " Gap in sequence number. Could not "
" seek to required sequence number " ) ;
" seek to required sequence number " ) ;
reporter_ . Info ( currentStatus_ . ToString ( ) . c_str ( ) ) ;
reporter_ . Info ( currentStatus_ . ToString ( ) . c_str ( ) ) ;
} else if ( files_ - > size ( ) ! = 1 ) {
} else {
currentStatus_ = Status : : Corruption ( " Start sequence was not found, "
currentStatus_ = Status : : Corruption ( " Start sequence was not found, "
" skipping to the next available " ) ;
" skipping to the next available " ) ;
reporter_ . Info ( currentStatus_ . ToString ( ) . c_str ( ) ) ;
reporter_ . Info ( currentStatus_ . ToString ( ) . c_str ( ) ) ;
@ -149,11 +164,30 @@ void TransactionLogIteratorImpl::NextImpl(bool internal) {
// Runs every time until we can seek to the start sequence
// Runs every time until we can seek to the start sequence
return SeekToStartSequence ( ) ;
return SeekToStartSequence ( ) ;
}
}
while ( true ) {
is_obsolete_ = false ;
auto latest_seq_num = dbimpl_ - > GetLatestSequenceNumber ( ) ;
if ( currentLastSeq_ > = latest_seq_num ) {
isValid_ = false ;
return ;
}
bool first = true ;
while ( currentFileIndex_ < files_ - > size ( ) ) {
if ( ! first ) {
Status status = OpenLogReader ( files_ - > at ( currentFileIndex_ ) . get ( ) ) ;
if ( ! status . ok ( ) ) {
isValid_ = false ;
currentStatus_ = status ;
return ;
}
}
first = false ;
assert ( currentLogReader_ ) ;
assert ( currentLogReader_ ) ;
if ( currentLogReader_ - > IsEOF ( ) ) {
if ( currentLogReader_ - > IsEOF ( ) ) {
currentLogReader_ - > UnmarkEOF ( ) ;
currentLogReader_ - > UnmarkEOF ( ) ;
}
}
while ( RestrictedRead ( & record , & scratch ) ) {
while ( RestrictedRead ( & record , & scratch ) ) {
if ( record . size ( ) < 12 ) {
if ( record . size ( ) < 12 ) {
reporter_ . Corruption (
reporter_ . Corruption (
@ -171,26 +205,14 @@ void TransactionLogIteratorImpl::NextImpl(bool internal) {
return ;
return ;
}
}
}
}
// Open the next file
// Open the next file
if ( currentFileIndex_ < files_ - > size ( ) - 1 ) {
+ + currentFileIndex_ ;
+ + currentFileIndex_ ;
Status status = OpenLogReader ( files_ - > at ( currentFileIndex_ ) . get ( ) ) ;
if ( ! status . ok ( ) ) {
isValid_ = false ;
currentStatus_ = status ;
return ;
}
}
} else {
// Read all the files but cannot find next record expected.
// TODO(sdong): support to auto fetch new log files from DB and continue.
isValid_ = false ;
isValid_ = false ;
if ( currentLastSeq_ = = dbimpl_ - > GetLatestSequenceNumber ( ) ) {
is_obsolete_ = true ;
currentStatus_ = Status : : OK ( ) ;
} else {
currentStatus_ = Status : : Corruption ( " NO MORE DATA LEFT " ) ;
}
return ;
}
}
}
}
bool TransactionLogIteratorImpl : : IsBatchExpected (
bool TransactionLogIteratorImpl : : IsBatchExpected (