2022-06-07 18:59:14 +00:00
package migrations
import (
"context"
2022-10-04 10:14:18 +00:00
"encoding/binary"
2022-06-07 18:59:14 +00:00
2022-11-20 03:41:30 +00:00
"github.com/ledgerwatch/erigon-lib/common/datadir"
2022-06-07 18:59:14 +00:00
"github.com/ledgerwatch/erigon-lib/kv"
2022-10-04 10:14:18 +00:00
"github.com/ledgerwatch/erigon/common"
2022-06-07 18:59:14 +00:00
"github.com/ledgerwatch/erigon/core/rawdb"
"github.com/ledgerwatch/erigon/core/rawdb/rawdbreset"
2022-10-04 10:14:18 +00:00
"github.com/ledgerwatch/erigon/eth/ethconfig"
2022-07-04 12:43:46 +00:00
"github.com/ledgerwatch/erigon/eth/stagedsync/stages"
2022-10-04 10:14:18 +00:00
"github.com/ledgerwatch/erigon/turbo/snapshotsync"
2022-06-07 18:59:14 +00:00
"github.com/ledgerwatch/erigon/turbo/snapshotsync/snap"
"github.com/ledgerwatch/log/v3"
)
2022-10-04 10:14:18 +00:00
var resetBlocks4 = Migration {
Name : "reset_blocks_4" ,
2022-06-07 18:59:14 +00:00
Up : func ( db kv . RwDB , dirs datadir . Dirs , progress [ ] byte , BeforeCommit Callback ) ( err error ) {
tx , err := db . BeginRw ( context . Background ( ) )
if err != nil {
return err
}
defer tx . Rollback ( )
enabled , err := snap . Enabled ( tx )
if err != nil {
return err
}
if ! enabled {
if err := BeforeCommit ( tx , nil , true ) ; err != nil {
return err
}
2022-07-18 07:38:01 +00:00
return tx . Commit ( )
2022-06-07 18:59:14 +00:00
}
2022-10-04 10:14:18 +00:00
// Detect whether the correction is required
snaps := snapshotsync . NewRoSnapshots ( ethconfig . Snapshot {
Enabled : true ,
KeepBlocks : true ,
Produce : false ,
} , dirs . Snap )
snaps . ReopenFolder ( )
var lastFound bool
var lastBlockNum , lastBaseTxNum , lastAmount uint64
if err := snaps . Bodies . View ( func ( sns [ ] * snapshotsync . BodySegment ) error {
// Take the last snapshot
if len ( sns ) == 0 {
return nil
}
sn := sns [ len ( sns ) - 1 ]
sn . Iterate ( func ( blockNum uint64 , baseTxNum uint64 , txAmount uint64 ) error {
lastBlockNum = blockNum
lastBaseTxNum = baseTxNum
lastAmount = txAmount
lastFound = true
return nil
} )
return nil
} ) ; err != nil {
return err
}
if ! lastFound {
2022-06-07 18:59:14 +00:00
if err := BeforeCommit ( tx , nil , true ) ; err != nil {
return err
}
2022-07-18 07:38:01 +00:00
return tx . Commit ( )
2022-06-07 18:59:14 +00:00
}
2022-10-04 10:14:18 +00:00
c , err := tx . Cursor ( kv . BlockBody )
2022-06-07 18:59:14 +00:00
if err != nil {
return err
}
2022-10-04 10:14:18 +00:00
defer c . Close ( )
var fixNeeded bool
for k , _ , err := c . First ( ) ; k != nil ; k , _ , err = c . Next ( ) {
if err != nil {
return err
}
blockNumber := binary . BigEndian . Uint64 ( k [ : 8 ] )
if blockNumber != lastBlockNum + 1 {
continue
}
blockHash := common . BytesToHash ( k [ 8 : ] )
var hash common . Hash
if hash , err = rawdb . ReadCanonicalHash ( tx , blockNumber ) ; err != nil {
return err
}
// ReadBody is not returning baseTxId which is written into the DB record, but that value + 1
_ , baseTxId , _ := rawdb . ReadBody ( tx , blockHash , blockNumber )
if hash != blockHash {
continue
}
if lastBaseTxNum + lastAmount + 1 != baseTxId {
log . Info ( "Fix required, last block in seg files" , "height" , lastBlockNum , "baseTxNum" , lastBaseTxNum , "txAmount" , lastAmount , "first txId in DB" , baseTxId , "expected" , lastBaseTxNum + lastAmount + 1 )
fixNeeded = true
}
}
if ! fixNeeded {
log . Info ( "Fix is not required" )
if err := BeforeCommit ( tx , nil , true ) ; err != nil {
return err
}
return tx . Commit ( )
}
2022-07-04 12:43:46 +00:00
headersProgress , _ := stages . GetStageProgress ( tx , stages . Headers )
if headersProgress > 0 {
log . Warn ( "NOTE: this migration will remove recent blocks (and senders) to fix several recent bugs. Your node will re-download last ~400K blocks, should not take very long" )
}
2022-06-07 18:59:14 +00:00
2022-10-05 10:54:54 +00:00
if err := rawdbreset . ResetBlocks ( tx , db , nil , nil , dirs . Tmp ) ; err != nil {
2022-06-07 18:59:14 +00:00
return err
}
if err := rawdbreset . ResetSenders ( tx ) ; err != nil {
return err
}
if err := rawdbreset . ResetTxLookup ( tx ) ; err != nil {
return err
}
// This migration is no-op, but it forces the migration mechanism to apply it and thus write the DB schema version info
if err := BeforeCommit ( tx , nil , true ) ; err != nil {
return err
}
return tx . Commit ( )
} ,
}