2020-03-11 15:54:09 +00:00
|
|
|
package stateless
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"os/signal"
|
|
|
|
"syscall"
|
|
|
|
"time"
|
|
|
|
|
2020-06-09 05:52:38 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
2020-03-11 15:54:09 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/common/changeset"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/consensus/ethash"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core"
|
2020-06-09 05:52:38 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/core/rawdb"
|
2020-03-11 15:54:09 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/core/state"
|
2020-06-09 05:52:38 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/core/types"
|
2020-03-11 15:54:09 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/core/vm"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/ethdb"
|
2020-03-25 20:18:46 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/log"
|
2020-03-11 15:54:09 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// CheckChangeSets re-executes historical transactions in read-only mode
|
|
|
|
// and checks that their outputs match the database ChangeSets.
|
2020-06-09 05:52:38 +00:00
|
|
|
func CheckChangeSets(genesis *core.Genesis, blockNum uint64, chaindata string, historyfile string, nocheck bool, writeReceipts bool) error {
|
2020-03-26 21:52:22 +00:00
|
|
|
if len(historyfile) == 0 {
|
|
|
|
historyfile = chaindata
|
|
|
|
}
|
|
|
|
|
2020-03-11 15:54:09 +00:00
|
|
|
startTime := time.Now()
|
|
|
|
sigs := make(chan os.Signal, 1)
|
|
|
|
interruptCh := make(chan bool, 1)
|
|
|
|
signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
<-sigs
|
|
|
|
interruptCh <- true
|
|
|
|
}()
|
|
|
|
|
2020-06-16 13:36:16 +00:00
|
|
|
chainDb := ethdb.MustOpen(chaindata)
|
2020-03-25 20:18:46 +00:00
|
|
|
defer chainDb.Close()
|
|
|
|
historyDb := chainDb
|
|
|
|
if chaindata != historyfile {
|
2020-06-16 13:36:16 +00:00
|
|
|
historyDb = ethdb.MustOpen(historyfile)
|
2020-03-25 20:18:46 +00:00
|
|
|
}
|
2020-03-11 15:54:09 +00:00
|
|
|
|
2020-03-30 18:56:53 +00:00
|
|
|
chainConfig := genesis.Config
|
2020-03-11 15:54:09 +00:00
|
|
|
engine := ethash.NewFaker()
|
|
|
|
vmConfig := vm.Config{}
|
2020-06-06 20:49:06 +00:00
|
|
|
bc, err := core.NewBlockChain(chainDb, nil, chainConfig, engine, vmConfig, nil, nil, nil)
|
2020-03-11 15:54:09 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
noOpWriter := state.NewNoopWriter()
|
|
|
|
|
|
|
|
interrupt := false
|
2020-06-09 05:52:38 +00:00
|
|
|
batch := chainDb.NewBatch()
|
2020-03-11 15:54:09 +00:00
|
|
|
for !interrupt {
|
|
|
|
block := bc.GetBlockByNumber(blockNum)
|
|
|
|
if block == nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2020-06-09 05:52:38 +00:00
|
|
|
dbstate := state.NewPlainDBState(historyDb.KV(), block.NumberU64()-1)
|
2020-03-11 15:54:09 +00:00
|
|
|
intraBlockState := state.New(dbstate)
|
2020-07-01 14:56:56 +00:00
|
|
|
csw := state.NewChangeSetWriterPlain(block.NumberU64() - 1)
|
2020-03-25 20:18:46 +00:00
|
|
|
var blockWriter state.StateWriter
|
|
|
|
if nocheck {
|
|
|
|
blockWriter = noOpWriter
|
|
|
|
} else {
|
|
|
|
blockWriter = csw
|
2020-03-11 15:54:09 +00:00
|
|
|
}
|
|
|
|
|
2020-06-09 05:52:38 +00:00
|
|
|
receipts, err1 := runBlock(intraBlockState, noOpWriter, blockWriter, chainConfig, bc, block)
|
|
|
|
if err1 != nil {
|
|
|
|
return err1
|
|
|
|
}
|
|
|
|
if chainConfig.IsByzantium(block.Number()) {
|
|
|
|
receiptSha := types.DeriveSha(receipts)
|
|
|
|
if receiptSha != block.Header().ReceiptHash {
|
|
|
|
return fmt.Errorf("mismatched receipt headers for block %d", block.NumberU64())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if writeReceipts {
|
|
|
|
rawdb.WriteReceipts(batch, block.Hash(), block.NumberU64(), receipts)
|
|
|
|
if batch.BatchSize() >= chainDb.IdealBatchSize() {
|
|
|
|
log.Info("Committing receipts", "up to block", block.NumberU64(), "batch size", common.StorageSize(batch.BatchSize()))
|
|
|
|
if _, err := batch.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2020-03-11 15:54:09 +00:00
|
|
|
}
|
|
|
|
|
2020-03-25 20:18:46 +00:00
|
|
|
if !nocheck {
|
2020-04-09 17:23:29 +00:00
|
|
|
accountChanges, err := csw.GetAccountChanges()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-03-26 21:52:22 +00:00
|
|
|
var expectedAccountChanges []byte
|
2020-06-28 06:10:27 +00:00
|
|
|
expectedAccountChanges, err = changeset.EncodeAccountsPlain(accountChanges)
|
2020-03-25 20:18:46 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-03-11 15:54:09 +00:00
|
|
|
|
2020-06-11 06:19:15 +00:00
|
|
|
dbAccountChanges, err := historyDb.GetChangeSetByBlock(false /* storage */, blockNum)
|
2020-03-11 15:54:09 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-03-25 20:18:46 +00:00
|
|
|
if !bytes.Equal(dbAccountChanges, expectedAccountChanges) {
|
2020-07-01 14:56:56 +00:00
|
|
|
fmt.Printf("Unexpected account changes in block %d\nIn the database: ======================\n", blockNum)
|
|
|
|
if err = changeset.AccountChangeSetPlainBytes(dbAccountChanges).Walk(func(k, v []byte) error {
|
|
|
|
fmt.Printf("0x%x: %x\n", k, v)
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fmt.Printf("Expected: ==========================\n")
|
|
|
|
if err = changeset.AccountChangeSetPlainBytes(expectedAccountChanges).Walk(func(k, v []byte) error {
|
|
|
|
fmt.Printf("0x%x %x\n", k, v)
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-03-25 20:18:46 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-04-09 17:23:29 +00:00
|
|
|
expectedStorageChanges, err := csw.GetStorageChanges()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-03-25 20:18:46 +00:00
|
|
|
expectedtorageSerialized := make([]byte, 0)
|
|
|
|
if expectedStorageChanges.Len() > 0 {
|
2020-06-28 06:10:27 +00:00
|
|
|
expectedtorageSerialized, err = changeset.EncodeStoragePlain(expectedStorageChanges)
|
2020-03-25 20:18:46 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2020-03-11 15:54:09 +00:00
|
|
|
|
2020-06-11 06:19:15 +00:00
|
|
|
dbStorageChanges, err := historyDb.GetChangeSetByBlock(true /* storage */, blockNum)
|
2020-03-25 20:18:46 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !bytes.Equal(dbStorageChanges, expectedtorageSerialized) {
|
2020-07-01 14:56:56 +00:00
|
|
|
fmt.Printf("Unexpected storage changes in block %d\nIn the database: ======================\n", blockNum)
|
|
|
|
if err = changeset.StorageChangeSetPlainBytes(dbStorageChanges).Walk(func(k, v []byte) error {
|
|
|
|
fmt.Printf("0x%x: [%x]\n", k, v)
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fmt.Printf("Expected: ==========================\n")
|
|
|
|
if err = changeset.StorageChangeSetPlainBytes(expectedtorageSerialized).Walk(func(k, v []byte) error {
|
|
|
|
fmt.Printf("0x%x: [%x]\n", k, v)
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-03-25 20:18:46 +00:00
|
|
|
return nil
|
|
|
|
}
|
2020-03-11 15:54:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
blockNum++
|
|
|
|
if blockNum%1000 == 0 {
|
2020-03-25 20:18:46 +00:00
|
|
|
log.Info("Checked", "blocks", blockNum)
|
2020-03-11 15:54:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check for interrupts
|
|
|
|
select {
|
|
|
|
case interrupt = <-interruptCh:
|
|
|
|
fmt.Println("interrupted, please wait for cleanup...")
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
}
|
2020-06-09 05:52:38 +00:00
|
|
|
if writeReceipts {
|
|
|
|
log.Info("Committing final receipts", "batch size", common.StorageSize(batch.BatchSize()))
|
|
|
|
if _, err := batch.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2020-03-25 20:18:46 +00:00
|
|
|
log.Info("Checked", "blocks", blockNum, "next time specify --block", blockNum, "duration", time.Since(startTime))
|
2020-03-11 15:54:09 +00:00
|
|
|
return nil
|
|
|
|
}
|