2022-01-14 16:02:44 +00:00
|
|
|
package commands
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"os/signal"
|
2022-02-12 13:33:09 +00:00
|
|
|
"path/filepath"
|
2022-01-14 16:02:44 +00:00
|
|
|
"syscall"
|
|
|
|
|
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
|
|
kv2 "github.com/ledgerwatch/erigon-lib/kv/mdbx"
|
|
|
|
"github.com/ledgerwatch/erigon/common"
|
2022-05-14 19:17:23 +00:00
|
|
|
"github.com/ledgerwatch/erigon/consensus/ethash"
|
2022-01-14 16:02:44 +00:00
|
|
|
"github.com/ledgerwatch/erigon/core"
|
|
|
|
"github.com/ledgerwatch/erigon/core/rawdb"
|
|
|
|
"github.com/ledgerwatch/erigon/core/state"
|
|
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
|
|
"github.com/ledgerwatch/erigon/core/vm"
|
|
|
|
"github.com/ledgerwatch/erigon/eth/stagedsync"
|
2022-08-19 02:35:43 +00:00
|
|
|
datadir2 "github.com/ledgerwatch/erigon/node/nodecfg/datadir"
|
2022-01-14 16:02:44 +00:00
|
|
|
"github.com/ledgerwatch/erigon/params"
|
|
|
|
"github.com/ledgerwatch/erigon/turbo/trie"
|
|
|
|
"github.com/ledgerwatch/log/v3"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
withBlock(stateRootCmd)
|
2022-02-22 17:39:48 +00:00
|
|
|
withDataDir(stateRootCmd)
|
2022-01-14 16:02:44 +00:00
|
|
|
rootCmd.AddCommand(stateRootCmd)
|
|
|
|
}
|
|
|
|
|
|
|
|
var stateRootCmd = &cobra.Command{
|
|
|
|
Use: "stateroot",
|
|
|
|
Short: "Exerimental command to re-execute blocks from beginning and compute state root",
|
|
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
|
|
logger := log.New()
|
|
|
|
return StateRoot(genesis, logger, block, datadir)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
func StateRoot(genesis *core.Genesis, logger log.Logger, blockNum uint64, datadir string) error {
|
|
|
|
sigs := make(chan os.Signal, 1)
|
|
|
|
interruptCh := make(chan bool, 1)
|
|
|
|
signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
<-sigs
|
|
|
|
interruptCh <- true
|
|
|
|
}()
|
2022-08-19 02:35:43 +00:00
|
|
|
dirs := datadir2.New(datadir)
|
|
|
|
historyDb, err := kv2.NewMDBX(logger).Path(dirs.Chaindata).Open()
|
2022-01-14 16:02:44 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer historyDb.Close()
|
|
|
|
ctx := context.Background()
|
|
|
|
historyTx, err1 := historyDb.BeginRo(ctx)
|
|
|
|
if err1 != nil {
|
|
|
|
return err1
|
|
|
|
}
|
|
|
|
defer historyTx.Rollback()
|
2022-02-12 13:33:09 +00:00
|
|
|
stateDbPath := filepath.Join(datadir, "staterootdb")
|
2022-01-14 16:02:44 +00:00
|
|
|
if _, err = os.Stat(stateDbPath); err != nil {
|
|
|
|
if !errors.Is(err, os.ErrNotExist) {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else if err = os.RemoveAll(stateDbPath); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
db, err2 := kv2.NewMDBX(logger).Path(stateDbPath).Open()
|
|
|
|
if err2 != nil {
|
|
|
|
return err2
|
|
|
|
}
|
|
|
|
defer db.Close()
|
|
|
|
chainConfig := genesis.Config
|
|
|
|
vmConfig := vm.Config{}
|
|
|
|
|
|
|
|
noOpWriter := state.NewNoopWriter()
|
|
|
|
interrupt := false
|
|
|
|
block := uint64(0)
|
|
|
|
var rwTx kv.RwTx
|
|
|
|
defer func() {
|
|
|
|
rwTx.Rollback()
|
|
|
|
}()
|
|
|
|
if rwTx, err = db.BeginRw(ctx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
_, genesisIbs, err4 := genesis.ToBlock()
|
|
|
|
if err4 != nil {
|
|
|
|
return err4
|
|
|
|
}
|
|
|
|
w := state.NewPlainStateWriter(rwTx, nil, 0)
|
2022-05-26 16:20:34 +00:00
|
|
|
if err = genesisIbs.CommitBlock(¶ms.Rules{}, w); err != nil {
|
2022-01-14 16:02:44 +00:00
|
|
|
return fmt.Errorf("cannot write state: %w", err)
|
|
|
|
}
|
|
|
|
if err = rwTx.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
var tx kv.Tx
|
|
|
|
defer func() {
|
|
|
|
if tx != nil {
|
|
|
|
tx.Rollback()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
for !interrupt {
|
|
|
|
block++
|
|
|
|
if block >= blockNum {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
blockHash, err := rawdb.ReadCanonicalHash(historyTx, block)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
var b *types.Block
|
|
|
|
b, _, err = rawdb.ReadBlockWithSenders(historyTx, blockHash, block)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if b == nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if tx, err = db.BeginRo(ctx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if rwTx, err = db.BeginRw(ctx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
w = state.NewPlainStateWriter(rwTx, nil, block)
|
|
|
|
r := state.NewPlainStateReader(tx)
|
|
|
|
intraBlockState := state.New(r)
|
|
|
|
getHeader := func(hash common.Hash, number uint64) *types.Header { return rawdb.ReadHeader(historyTx, hash, number) }
|
2022-09-01 18:49:29 +00:00
|
|
|
if _, err = runBlock(ethash.NewFullFaker(), intraBlockState, noOpWriter, w, chainConfig, getHeader, b, vmConfig, false); err != nil {
|
2022-01-14 16:02:44 +00:00
|
|
|
return fmt.Errorf("block %d: %w", block, err)
|
|
|
|
}
|
|
|
|
if block+1 == blockNum {
|
|
|
|
if err = rwTx.ClearBucket(kv.HashedAccounts); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err = rwTx.ClearBucket(kv.HashedStorage); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-09-18 10:41:01 +00:00
|
|
|
if err = stagedsync.PromoteHashedStateCleanly("hashedstate", rwTx, stagedsync.StageHashStateCfg(nil, dirs, false, nil), ctx); err != nil {
|
2022-01-14 16:02:44 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
var root common.Hash
|
|
|
|
root, err = trie.CalcRoot("genesis", rwTx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fmt.Printf("root for block %d=[%x]\n", block, root)
|
|
|
|
}
|
|
|
|
if block%1000 == 0 {
|
|
|
|
log.Info("Processed", "blocks", block)
|
|
|
|
}
|
|
|
|
// Check for interrupts
|
|
|
|
select {
|
|
|
|
case interrupt = <-interruptCh:
|
|
|
|
fmt.Println("interrupted, please wait for cleanup...")
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
tx.Rollback()
|
|
|
|
if err = rwTx.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|