2019-05-27 13:51:49 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2019-10-21 12:00:42 +00:00
|
|
|
"encoding/binary"
|
2019-05-27 13:51:49 +00:00
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ledgerwatch/bolt"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/common/dbutils"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/consensus/ethash"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core/state"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core/types/accounts"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core/vm"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/crypto"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/ethdb"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/params"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/trie"
|
|
|
|
)
|
|
|
|
|
|
|
|
// NOTE: This file is not the part of the Turbo-Geth binary. It i s part of the experimental utility, state
|
|
|
|
// to perform data analysis related to the state growth, state rent, and statelesss clients
|
|
|
|
|
|
|
|
func constructSnapshot(ethDb ethdb.Database, blockNum uint64) {
|
|
|
|
diskDb, err := bolt.Open(fmt.Sprintf("/Volumes/tb4/turbo-geth-copy/state_%d", blockNum), 0600, &bolt.Options{})
|
|
|
|
check(err)
|
|
|
|
defer diskDb.Close()
|
|
|
|
var startKey [32]byte
|
|
|
|
txDisk, err := diskDb.Begin(true)
|
|
|
|
check(err)
|
|
|
|
bDisk, err := txDisk.CreateBucket(dbutils.AccountsBucket, true)
|
|
|
|
check(err)
|
|
|
|
count := 0
|
|
|
|
err = ethDb.WalkAsOf(dbutils.AccountsBucket, dbutils.AccountsHistoryBucket, startKey[:], 0, blockNum+1,
|
|
|
|
func(key []byte, value []byte) (bool, error) {
|
|
|
|
if len(value) == 0 {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
if err = bDisk.Put(common.CopyBytes(key), common.CopyBytes(value)); err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
count++
|
|
|
|
if count%1000 == 0 {
|
|
|
|
if err := txDisk.Commit(); err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
fmt.Printf("Committed %d records\n", count)
|
|
|
|
var err error
|
|
|
|
txDisk, err = diskDb.Begin(true)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
bDisk = txDisk.Bucket(dbutils.AccountsBucket)
|
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
},
|
|
|
|
)
|
|
|
|
check(err)
|
|
|
|
err = txDisk.Commit()
|
|
|
|
check(err)
|
|
|
|
txDisk, err = diskDb.Begin(true)
|
|
|
|
check(err)
|
|
|
|
b := txDisk.Bucket(dbutils.AccountsBucket)
|
|
|
|
sbDisk, err := txDisk.CreateBucket(dbutils.StorageBucket, true)
|
|
|
|
check(err)
|
|
|
|
count = 0
|
|
|
|
var address common.Address
|
|
|
|
//var hash common.Hash
|
|
|
|
exist := make(map[common.Address]bool)
|
|
|
|
var sk [52]byte
|
|
|
|
err = ethDb.WalkAsOf(dbutils.StorageBucket, dbutils.StorageHistoryBucket, sk[:], 0, blockNum,
|
|
|
|
func(key []byte, value []byte) (bool, error) {
|
|
|
|
if len(value) == 0 {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
copy(address[:], key[:20])
|
|
|
|
if e, ok := exist[address]; ok {
|
|
|
|
if !e {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
v, _ := b.Get(crypto.Keccak256(address[:]))
|
|
|
|
exist[address] = v != nil
|
|
|
|
}
|
|
|
|
if err = sbDisk.Put(common.CopyBytes(key), common.CopyBytes(value)); err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
count++
|
|
|
|
if count%1000 == 0 {
|
|
|
|
if err := txDisk.Commit(); err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
fmt.Printf("Committed %d records\n", count)
|
|
|
|
var err error
|
|
|
|
txDisk, err = diskDb.Begin(true)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
b = txDisk.Bucket(dbutils.AccountsBucket)
|
|
|
|
sbDisk = txDisk.Bucket(dbutils.StorageBucket)
|
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
},
|
|
|
|
)
|
|
|
|
check(err)
|
|
|
|
err = txDisk.Commit()
|
|
|
|
check(err)
|
|
|
|
}
|
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
type bucketWriter struct {
|
|
|
|
db ethdb.Database
|
|
|
|
bucket []byte
|
|
|
|
pending ethdb.DbWithPendingMutations
|
|
|
|
written uint64
|
|
|
|
}
|
2019-11-21 13:36:24 +00:00
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
func (bw *bucketWriter) printStats() {
|
|
|
|
if bw.written == 0 {
|
|
|
|
fmt.Printf(" -- nothing to copy for bucket: '%s'...", string(bw.bucket))
|
|
|
|
} else {
|
|
|
|
fmt.Printf("\r -- commited %d records for bucket: '%s'...", bw.written, string(bw.bucket))
|
2019-11-21 13:36:24 +00:00
|
|
|
}
|
2019-11-21 13:57:44 +00:00
|
|
|
}
|
2019-11-21 13:36:24 +00:00
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
func (bw *bucketWriter) walker(k, v []byte) (bool, error) {
|
|
|
|
if bw.pending == nil {
|
|
|
|
bw.pending = bw.db.NewBatch()
|
|
|
|
}
|
2019-11-21 13:36:24 +00:00
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
if err := bw.pending.Put(bw.bucket, common.CopyBytes(k), common.CopyBytes(v)); err != nil {
|
|
|
|
return false, err
|
2019-11-21 13:36:24 +00:00
|
|
|
}
|
2019-11-21 13:57:44 +00:00
|
|
|
bw.written++
|
2019-11-21 13:36:24 +00:00
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
if bw.pending.BatchSize() >= 100000 {
|
|
|
|
if _, err := bw.pending.Commit(); err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
2019-11-21 13:36:24 +00:00
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
bw.printStats()
|
2019-11-21 13:36:24 +00:00
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
bw.pending = bw.db.NewBatch()
|
|
|
|
}
|
2019-11-21 13:36:24 +00:00
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
return true, nil
|
|
|
|
}
|
2019-11-21 13:36:24 +00:00
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
func (bw *bucketWriter) commit() error {
|
|
|
|
defer bw.printStats()
|
2019-11-21 13:36:24 +00:00
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
if bw.pending != nil {
|
|
|
|
_, err := bw.pending.Commit()
|
|
|
|
return err
|
2019-11-21 13:36:24 +00:00
|
|
|
}
|
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
return nil
|
2019-11-21 13:36:24 +00:00
|
|
|
}
|
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
func newBucketWriter(db ethdb.Database, bucket []byte) *bucketWriter {
|
|
|
|
return &bucketWriter{
|
|
|
|
db: db,
|
|
|
|
bucket: bucket,
|
|
|
|
pending: nil,
|
|
|
|
written: 0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func copyDatabase(fromDB ethdb.Database, toDB ethdb.Database) error {
|
|
|
|
for _, bucket := range [][]byte{dbutils.AccountsBucket, dbutils.StorageBucket, dbutils.CodeBucket} {
|
|
|
|
fmt.Printf(" - copying bucket '%s'...\n", string(bucket))
|
|
|
|
writer := newBucketWriter(toDB, bucket)
|
2019-11-21 13:36:24 +00:00
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
if err := fromDB.Walk(bucket, nil, 0, writer.walker); err != nil {
|
2019-11-21 13:36:24 +00:00
|
|
|
fmt.Println("FAIL")
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
if err := writer.commit(); err != nil {
|
2019-11-21 13:36:24 +00:00
|
|
|
fmt.Println("FAIL")
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fmt.Println("OK")
|
2019-11-21 13:57:44 +00:00
|
|
|
}
|
|
|
|
return nil
|
2019-11-21 13:36:24 +00:00
|
|
|
}
|
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
func saveSnapshot(db ethdb.Database, filename string, createDb CreateDbFunc) {
|
2019-11-21 13:36:24 +00:00
|
|
|
fmt.Printf("Saving snapshot to %s\n", filename)
|
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
diskDb, err := createDb(filename)
|
2019-05-27 13:51:49 +00:00
|
|
|
check(err)
|
2019-11-21 13:36:24 +00:00
|
|
|
defer diskDb.Close()
|
|
|
|
|
|
|
|
err = copyDatabase(db, diskDb)
|
2019-05-27 13:51:49 +00:00
|
|
|
check(err)
|
|
|
|
}
|
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
func loadSnapshot(db ethdb.Database, filename string, createDb CreateDbFunc) {
|
2019-05-27 13:51:49 +00:00
|
|
|
fmt.Printf("Loading snapshot from %s\n", filename)
|
2019-11-21 13:57:44 +00:00
|
|
|
|
|
|
|
diskDb, err := createDb(filename)
|
2019-05-27 13:51:49 +00:00
|
|
|
check(err)
|
2019-11-21 13:36:24 +00:00
|
|
|
defer diskDb.Close()
|
|
|
|
|
|
|
|
err = copyDatabase(diskDb, db)
|
2019-05-27 13:51:49 +00:00
|
|
|
check(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func loadCodes(db *bolt.DB, codeDb ethdb.Database) error {
|
|
|
|
var account accounts.Account
|
|
|
|
err := db.Update(func(tx *bolt.Tx) error {
|
|
|
|
b := tx.Bucket(dbutils.AccountsBucket)
|
|
|
|
cb, err := tx.CreateBucket(dbutils.CodeBucket, true)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c := b.Cursor()
|
|
|
|
for k, v := c.First(); k != nil; k, v = c.Next() {
|
|
|
|
if err := account.DecodeForStorage(v); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !account.IsEmptyCodeHash() {
|
|
|
|
code, _ := codeDb.Get(dbutils.CodeBucket, account.CodeHash[:])
|
|
|
|
if code != nil {
|
|
|
|
if err := cb.Put(account.CodeHash[:], code); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func compare_snapshot(stateDb ethdb.Database, db *bolt.DB, filename string) {
|
|
|
|
fmt.Printf("Loading snapshot from %s\n", filename)
|
|
|
|
diskDb, err := bolt.Open(filename, 0600, &bolt.Options{})
|
|
|
|
check(err)
|
|
|
|
defer diskDb.Close()
|
|
|
|
err = db.View(func(tx *bolt.Tx) error {
|
|
|
|
b := tx.Bucket(dbutils.AccountsBucket)
|
|
|
|
sb := tx.Bucket(dbutils.StorageBucket)
|
|
|
|
preimage := tx.Bucket(dbutils.PreimagePrefix)
|
|
|
|
count := 0
|
|
|
|
err = diskDb.View(func(txDisk *bolt.Tx) error {
|
|
|
|
bDisk := txDisk.Bucket(dbutils.AccountsBucket)
|
|
|
|
cDisk := bDisk.Cursor()
|
|
|
|
for k, v := cDisk.First(); k != nil; k, v = cDisk.Next() {
|
|
|
|
vv, _ := b.Get(k)
|
|
|
|
p, _ := preimage.Get(k)
|
|
|
|
if !bytes.Equal(v, vv) {
|
|
|
|
fmt.Printf("Diff for %x (%x): disk: %x, mem: %x\n", k, p, v, vv)
|
|
|
|
}
|
|
|
|
count++
|
|
|
|
if count%100000 == 0 {
|
|
|
|
fmt.Printf("Compared %d records\n", count)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sbDisk := txDisk.Bucket(dbutils.StorageBucket)
|
|
|
|
count = 0
|
|
|
|
cDisk = sbDisk.Cursor()
|
|
|
|
for k, v := cDisk.First(); k != nil; k, v = cDisk.Next() {
|
|
|
|
vv, _ := sb.Get(k)
|
|
|
|
if !bytes.Equal(v, vv) {
|
|
|
|
fmt.Printf("Diff for %x: disk: %x, mem: %x\n", k, v, vv)
|
|
|
|
}
|
|
|
|
count++
|
|
|
|
if count%100000 == 0 {
|
|
|
|
fmt.Printf("Committed %d records\n", count)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
count = 0
|
|
|
|
c := b.Cursor()
|
|
|
|
for k, v := c.First(); k != nil; k, v = c.Next() {
|
|
|
|
vv, _ := bDisk.Get(k)
|
|
|
|
p, _ := preimage.Get(k)
|
|
|
|
if len(vv) == 0 {
|
|
|
|
fmt.Printf("Diff for %x (%x): disk: %x, mem: %x\n", k, p, vv, v)
|
|
|
|
}
|
|
|
|
count++
|
|
|
|
if count%100000 == 0 {
|
|
|
|
fmt.Printf("Compared %d records\n", count)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c = sb.Cursor()
|
|
|
|
for k, v := c.First(); k != nil; k, v = c.Next() {
|
|
|
|
vv, _ := sbDisk.Get(k)
|
|
|
|
p, _ := preimage.Get(k)
|
|
|
|
if len(vv) == 0 {
|
|
|
|
fmt.Printf("Diff for %x (%x): disk: %x, mem: %x\n", k, p, vv, v)
|
|
|
|
}
|
|
|
|
count++
|
|
|
|
if count%100000 == 0 {
|
|
|
|
fmt.Printf("Compared %d records\n", count)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
check(err)
|
|
|
|
}
|
|
|
|
|
2019-11-21 13:57:44 +00:00
|
|
|
func checkRoots(stateDb ethdb.Database, rootHash common.Hash, blockNum uint64) {
|
2019-05-27 13:51:49 +00:00
|
|
|
startTime := time.Now()
|
|
|
|
t := trie.New(rootHash)
|
|
|
|
r := trie.NewResolver(0, true, blockNum)
|
|
|
|
key := []byte{}
|
|
|
|
req := t.NewResolveRequest(nil, key, 0, rootHash[:])
|
2019-11-21 13:36:24 +00:00
|
|
|
fmt.Printf("new resolve request for root block with hash %x\n", rootHash)
|
2019-05-27 13:51:49 +00:00
|
|
|
r.AddRequest(req)
|
2019-10-21 12:00:42 +00:00
|
|
|
var err error
|
|
|
|
if err = r.ResolveWithDb(stateDb, blockNum); err != nil {
|
2019-05-27 13:51:49 +00:00
|
|
|
fmt.Printf("%v\n", err)
|
|
|
|
}
|
|
|
|
fmt.Printf("Trie computation took %v\n", time.Since(startTime))
|
|
|
|
startTime = time.Now()
|
2019-11-25 13:36:21 +00:00
|
|
|
roots := make(map[common.Hash]*accounts.Account)
|
2019-11-21 13:57:44 +00:00
|
|
|
err = stateDb.Walk(dbutils.StorageBucket, nil, 0, func(k, v []byte) (bool, error) {
|
2019-11-25 13:36:21 +00:00
|
|
|
var addrHash common.Hash
|
2019-11-21 13:57:44 +00:00
|
|
|
copy(addrHash[:], k[:32])
|
|
|
|
if _, ok := roots[addrHash]; !ok {
|
|
|
|
if enc, _ := stateDb.Get(dbutils.AccountsBucket, addrHash[:]); enc == nil {
|
2019-11-25 13:36:21 +00:00
|
|
|
roots[addrHash] = nil
|
2019-11-21 13:57:44 +00:00
|
|
|
} else {
|
|
|
|
var account accounts.Account
|
|
|
|
if err = account.DecodeForStorage(enc); err != nil {
|
|
|
|
return false, err
|
2019-05-27 13:51:49 +00:00
|
|
|
}
|
2019-11-25 13:36:21 +00:00
|
|
|
roots[addrHash] = &account
|
2019-05-27 13:51:49 +00:00
|
|
|
}
|
|
|
|
}
|
2019-11-21 13:57:44 +00:00
|
|
|
|
|
|
|
return true, nil
|
2019-05-27 13:51:49 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2019-11-25 13:36:21 +00:00
|
|
|
for addrHash, account := range roots {
|
|
|
|
if account != nil {
|
|
|
|
st := trie.New(account.Root)
|
2019-05-27 13:51:49 +00:00
|
|
|
sr := trie.NewResolver(32, false, blockNum)
|
|
|
|
key := []byte{}
|
|
|
|
contractPrefix := make([]byte, common.HashLength+state.IncarnationLength)
|
2019-10-21 12:00:42 +00:00
|
|
|
copy(contractPrefix, addrHash[:])
|
2019-11-25 13:36:21 +00:00
|
|
|
binary.BigEndian.PutUint64(contractPrefix[common.HashLength:], account.Incarnation^^uint64(0))
|
|
|
|
streq := st.NewResolveRequest(contractPrefix, key, 0, account.Root[:])
|
2019-05-27 13:51:49 +00:00
|
|
|
sr.AddRequest(streq)
|
|
|
|
err = sr.ResolveWithDb(stateDb, blockNum)
|
|
|
|
if err != nil {
|
2019-10-21 12:00:42 +00:00
|
|
|
fmt.Printf("%x: %v\n", addrHash, err)
|
|
|
|
filename := fmt.Sprintf("tries/root_%x.txt", addrHash)
|
2019-05-27 13:51:49 +00:00
|
|
|
f, err := os.Create(filename)
|
|
|
|
if err == nil {
|
|
|
|
defer f.Close()
|
|
|
|
st.Print(f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fmt.Printf("Storage trie computation took %v\n", time.Since(startTime))
|
|
|
|
}
|
|
|
|
|
|
|
|
func stateSnapshot() error {
|
|
|
|
startTime := time.Now()
|
|
|
|
var blockNum uint64 = uint64(*block)
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("/Users/alexeyakhunov/Library/Ethereum/geth/chaindata")
|
|
|
|
ethDb, err := ethdb.NewBoltDatabase("/Volumes/tb4/turbo-geth-copy/geth/chaindata")
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("/home/akhounov/.ethereum/geth/chaindata1")
|
|
|
|
check(err)
|
|
|
|
defer ethDb.Close()
|
|
|
|
stateDb, db := ethdb.NewMemDatabase2()
|
|
|
|
defer stateDb.Close()
|
|
|
|
if _, err := os.Stat("statedb0"); err == nil {
|
2019-11-21 13:57:44 +00:00
|
|
|
loadSnapshot(stateDb, "statedb0", func(path string) (ethdb.Database, error) {
|
|
|
|
return ethdb.NewBoltDatabase(path)
|
|
|
|
})
|
2019-05-27 13:51:49 +00:00
|
|
|
if err := loadCodes(db, ethDb); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
constructSnapshot(ethDb, blockNum)
|
|
|
|
}
|
|
|
|
fmt.Printf("Snapshot took %v\n", time.Since(startTime))
|
|
|
|
startTime = time.Now()
|
|
|
|
bc, err := core.NewBlockChain(ethDb, nil, params.MainnetChainConfig, ethash.NewFaker(), vm.Config{}, nil)
|
|
|
|
check(err)
|
|
|
|
block := bc.GetBlockByNumber(blockNum)
|
|
|
|
fmt.Printf("Block number: %d\n", blockNum)
|
|
|
|
fmt.Printf("Block root hash: %x\n", block.Root())
|
2019-11-21 13:57:44 +00:00
|
|
|
checkRoots(ethDb, block.Root(), blockNum)
|
2019-05-27 13:51:49 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-10-21 12:00:42 +00:00
|
|
|
func verifySnapshot(chaindata string) {
|
2019-05-27 13:51:49 +00:00
|
|
|
blockNum := uint64(*block)
|
2019-10-21 12:00:42 +00:00
|
|
|
ethDb, err := ethdb.NewBoltDatabase(chaindata)
|
2019-05-27 13:51:49 +00:00
|
|
|
check(err)
|
|
|
|
defer ethDb.Close()
|
|
|
|
engine := ethash.NewFullFaker()
|
|
|
|
chainConfig := params.MainnetChainConfig
|
2019-10-21 12:00:42 +00:00
|
|
|
bc, err := core.NewBlockChain(ethDb, nil, chainConfig, engine, vm.Config{}, nil)
|
2019-05-27 13:51:49 +00:00
|
|
|
check(err)
|
2019-10-21 12:00:42 +00:00
|
|
|
currentBlock := bc.CurrentBlock()
|
|
|
|
currentBlockNr := currentBlock.NumberU64()
|
|
|
|
fmt.Printf("Block number: %d\n", currentBlockNr)
|
|
|
|
fmt.Printf("Block root hash: %x\n", currentBlock.Root())
|
|
|
|
preRoot := currentBlock.Root()
|
2019-11-21 13:57:44 +00:00
|
|
|
checkRoots(ethDb, preRoot, blockNum)
|
2019-05-27 13:51:49 +00:00
|
|
|
}
|