package commands import ( "context" "fmt" "os" "text/tabwriter" "github.com/ledgerwatch/turbo-geth/cmd/utils" "github.com/ledgerwatch/turbo-geth/common/dbutils" "github.com/ledgerwatch/turbo-geth/core" "github.com/ledgerwatch/turbo-geth/eth/stagedsync" "github.com/ledgerwatch/turbo-geth/eth/stagedsync/stages" "github.com/ledgerwatch/turbo-geth/ethdb" "github.com/ledgerwatch/turbo-geth/log" "github.com/spf13/cobra" ) var cmdResetState = &cobra.Command{ Use: "reset_state", Short: "Reset StateStages (5,6,7,8,9,10) and buckets", RunE: func(cmd *cobra.Command, args []string) error { ctx := utils.RootContext() db := openDatabase(chaindata, true) defer db.Close() err := resetState(db, ctx) if err != nil { log.Error(err.Error()) return err } return nil }, } var cmdClearUnwindStack = &cobra.Command{ Use: "clear_unwind_stack", Short: "Clear unwind stack", RunE: func(cmd *cobra.Command, args []string) error { ctx := utils.RootContext() db := openDatabase(chaindata, true) defer db.Close() err := clearUnwindStack(db, ctx) if err != nil { log.Error(err.Error()) return err } return nil }, } func init() { withChaindata(cmdResetState) rootCmd.AddCommand(cmdResetState) withChaindata(cmdClearUnwindStack) rootCmd.AddCommand(cmdClearUnwindStack) } func clearUnwindStack(db ethdb.Putter, _ context.Context) error { for _, stage := range stages.AllStages { if err := stages.SaveStageUnwind(db, stage, 0); err != nil { return err } } return nil } func resetState(db ethdb.Database, ctx context.Context) error { fmt.Printf("Before reset: \n") if err := printStages(db); err != nil { return err } kv := db.(ethdb.HasRwKV).RwKV() // don't reset senders here if err := kv.Update(ctx, func(tx ethdb.RwTx) error { return resetExec(tx) }); err != nil { return err } if err := kv.Update(ctx, func(tx ethdb.RwTx) error { return stagedsync.ResetHashState(tx) }); err != nil { return err } if err := kv.Update(ctx, func(tx ethdb.RwTx) error { return stagedsync.ResetIH(tx) }); err != nil { return err } if err := kv.Update(ctx, func(tx ethdb.RwTx) error { return resetHistory(tx) }); err != nil { return err } if err := kv.Update(ctx, func(tx ethdb.RwTx) error { return resetLogIndex(tx) }); err != nil { return err } if err := kv.Update(ctx, func(tx ethdb.RwTx) error { return resetCallTraces(tx) }); err != nil { return err } if err := kv.Update(ctx, func(tx ethdb.RwTx) error { return resetTxLookup(tx) }); err != nil { return err } if err := kv.Update(ctx, func(tx ethdb.RwTx) error { return resetTxPool(tx) }); err != nil { return err } if err := kv.Update(ctx, func(tx ethdb.RwTx) error { return resetFinish(tx) }); err != nil { return err } // set genesis after reset all buckets tx, err := db.Begin(context.Background(), ethdb.RW) if err != nil { return err } if _, _, err = core.DefaultGenesisBlock().WriteGenesisState(tx, false); err != nil { return err } if err := tx.Commit(); err != nil { return err } fmt.Printf("After reset: \n") if err := printStages(db); err != nil { return err } return nil } func resetSenders(tx ethdb.RwTx) error { if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.Senders); err != nil { return err } if err := stages.SaveStageProgress(tx, stages.Senders, 0); err != nil { return err } if err := stages.SaveStageUnwind(tx, stages.Senders, 0); err != nil { return err } return nil } func resetExec(tx ethdb.RwTx) error { if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.HashedAccountsBucket); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.HashedStorageBucket); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.ContractCodeBucket); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.PlainStateBucket); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.PlainAccountChangeSetBucket); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.PlainStorageChangeSetBucket); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.PlainContractCodeBucket); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.BlockReceiptsPrefix); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.Log); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.IncarnationMapBucket); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.CodeBucket); err != nil { return err } if err := stages.SaveStageProgress(tx, stages.Execution, 0); err != nil { return err } if err := stages.SaveStageUnwind(tx, stages.Execution, 0); err != nil { return err } return nil } func resetHistory(tx ethdb.RwTx) error { if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.AccountsHistoryBucket); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.StorageHistoryBucket); err != nil { return err } if err := stages.SaveStageProgress(tx, stages.AccountHistoryIndex, 0); err != nil { return err } if err := stages.SaveStageProgress(tx, stages.StorageHistoryIndex, 0); err != nil { return err } if err := stages.SaveStageUnwind(tx, stages.AccountHistoryIndex, 0); err != nil { return err } if err := stages.SaveStageUnwind(tx, stages.StorageHistoryIndex, 0); err != nil { return err } return nil } func resetLogIndex(tx ethdb.RwTx) error { if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.LogAddressIndex); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.LogTopicIndex); err != nil { return err } if err := stages.SaveStageProgress(tx, stages.LogIndex, 0); err != nil { return err } if err := stages.SaveStageUnwind(tx, stages.LogIndex, 0); err != nil { return err } return nil } func resetCallTraces(tx ethdb.RwTx) error { if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.CallFromIndex); err != nil { return err } if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.CallToIndex); err != nil { return err } if err := stages.SaveStageProgress(tx, stages.CallTraces, 0); err != nil { return err } if err := stages.SaveStageUnwind(tx, stages.CallTraces, 0); err != nil { return err } return nil } func resetTxLookup(tx ethdb.RwTx) error { if err := tx.(ethdb.BucketMigrator).ClearBucket(dbutils.TxLookupPrefix); err != nil { return err } if err := stages.SaveStageProgress(tx, stages.TxLookup, 0); err != nil { return err } if err := stages.SaveStageUnwind(tx, stages.TxLookup, 0); err != nil { return err } return nil } func resetTxPool(tx ethdb.RwTx) error { if err := stages.SaveStageProgress(tx, stages.TxPool, 0); err != nil { return err } if err := stages.SaveStageUnwind(tx, stages.TxPool, 0); err != nil { return err } return nil } func resetFinish(tx ethdb.RwTx) error { if err := stages.SaveStageProgress(tx, stages.Finish, 0); err != nil { return err } if err := stages.SaveStageUnwind(tx, stages.Finish, 0); err != nil { return err } return nil } func printStages(db ethdb.Getter) error { var err error var progress uint64 w := new(tabwriter.Writer) defer w.Flush() w.Init(os.Stdout, 8, 8, 0, '\t', 0) for _, stage := range stages.AllStages { if progress, err = stages.GetStageProgress(db, stage); err != nil { return err } fmt.Fprintf(w, "%s \t %d\n", string(stage), progress) } return nil }