erigon-pulse/cmd/snapshots/generator/commands/generate_state_snapshot.go

164 lines
4.6 KiB
Go

package commands
import (
"context"
"errors"
"fmt"
"os"
"time"
"github.com/ledgerwatch/erigon/common"
"github.com/ledgerwatch/erigon/common/dbutils"
"github.com/ledgerwatch/erigon/core/state"
"github.com/ledgerwatch/erigon/core/types/accounts"
"github.com/ledgerwatch/erigon/ethdb"
"github.com/ledgerwatch/erigon/ethdb/kv"
kv2 "github.com/ledgerwatch/erigon/ethdb/mdbx"
"github.com/ledgerwatch/erigon/log"
"github.com/ledgerwatch/erigon/turbo/trie"
"github.com/spf13/cobra"
)
func init() {
withDatadir(generateStateSnapshotCmd)
withSnapshotFile(generateStateSnapshotCmd)
withBlock(generateStateSnapshotCmd)
rootCmd.AddCommand(generateStateSnapshotCmd)
}
//go run cmd/snapshots/generator/main.go state_copy --block 11000000 --snapshot /media/b00ris/nvme/snapshots/state --datadir /media/b00ris/nvme/backup/snapshotsync &> /media/b00ris/nvme/copy.log
var generateStateSnapshotCmd = &cobra.Command{
Use: "state",
Short: "Generate state snapshot",
Example: "go run ./cmd/state/main.go stateSnapshot --block 11000000 --datadir /media/b00ris/nvme/tgstaged/ --snapshot /media/b00ris/nvme/snapshots/state",
RunE: func(cmd *cobra.Command, args []string) error {
return GenerateStateSnapshot(cmd.Context(), log.New(), chaindata, snapshotFile, block, snapshotDir, snapshotMode)
},
}
func GenerateStateSnapshot(ctx context.Context, logger log.Logger, dbPath, snapshotPath string, toBlock uint64, snapshotDir string, snapshotMode string) error {
if snapshotPath == "" {
return errors.New("empty snapshot path")
}
err := os.RemoveAll(snapshotPath)
if err != nil {
return err
}
var db, snkv kv.RwDB
db = kv2.NewMDBX(logger).Path(dbPath).MustOpen()
snkv = kv2.NewMDBX(logger).WithBucketsConfig(func(defaultBuckets kv.TableCfg) kv.TableCfg {
return kv.TableCfg{
kv.PlainStateBucket: kv.TableConfigItem{},
kv.PlainContractCode: kv.TableConfigItem{},
kv.CodeBucket: kv.TableConfigItem{},
}
}).Path(snapshotPath).MustOpen()
writeTx, err := snkv.BeginRw(ctx)
if err != nil {
return err
}
defer writeTx.Rollback()
tx, err := db.BeginRo(context.Background())
if err != nil {
return err
}
tx2, err := db.BeginRo(context.Background())
if err != nil {
return err
}
defer tx.Rollback()
i := 0
t := time.Now()
tt := time.Now()
commitEvery := time.NewTicker(30 * time.Second)
defer commitEvery.Stop()
err = state.WalkAsOfAccounts(tx, common.Address{}, toBlock+1, func(k []byte, v []byte) (bool, error) {
i++
if i%100000 == 0 {
fmt.Println(i, common.Bytes2Hex(k), "batch", time.Since(tt))
tt = time.Now()
select {
case <-ctx.Done():
return false, errors.New("interrupted")
default:
}
}
if len(k) != 20 {
return true, nil
}
var acc accounts.Account
if err = acc.DecodeForStorage(v); err != nil {
return false, fmt.Errorf("decoding %x for %x: %v", v, k, err)
}
if acc.Incarnation > 0 {
storagePrefix := dbutils.PlainGenerateStoragePrefix(k, acc.Incarnation)
if acc.IsEmptyRoot() {
t := trie.New(common.Hash{})
j := 0
innerErr := state.WalkAsOfStorage(tx2, common.BytesToAddress(k), acc.Incarnation, common.Hash{}, toBlock+1, func(k1, k2 []byte, vv []byte) (bool, error) {
j++
innerErr1 := writeTx.Put(kv.PlainStateBucket, dbutils.PlainGenerateCompositeStorageKey(k1, acc.Incarnation, k2), common.CopyBytes(vv))
if innerErr1 != nil {
return false, innerErr1
}
h, _ := common.HashData(k1)
t.Update(h.Bytes(), common.CopyBytes(vv))
return true, nil
})
if innerErr != nil {
return false, innerErr
}
acc.Root = t.Hash()
}
if acc.IsEmptyCodeHash() {
codeHash, err1 := tx2.GetOne(kv.PlainContractCode, storagePrefix)
if err1 != nil && errors.Is(err1, ethdb.ErrKeyNotFound) {
return false, fmt.Errorf("getting code hash for %x: %v", k, err1)
}
if len(codeHash) > 0 {
code, err1 := tx2.GetOne(kv.CodeBucket, codeHash)
if err1 != nil {
return false, err1
}
if err1 = writeTx.Put(kv.CodeBucket, codeHash, code); err1 != nil {
return false, err1
}
if err1 = writeTx.Put(kv.PlainContractCode, storagePrefix, codeHash); err1 != nil {
return false, err1
}
}
}
}
newAcc := make([]byte, acc.EncodingLengthForStorage())
acc.EncodeForStorage(newAcc)
innerErr := writeTx.Put(kv.PlainStateBucket, common.CopyBytes(k), newAcc)
if innerErr != nil {
return false, innerErr
}
return true, nil
})
if err != nil {
return err
}
err = writeTx.Commit()
if err != nil {
return err
}
fmt.Println("took", time.Since(t))
return VerifyStateSnapshot(ctx, logger, dbPath, snapshotFile, block)
}