| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529 |
- // Copyright 2020 The go-ethereum Authors
- // This file is part of go-ethereum.
- //
- // go-ethereum is free software: you can redistribute it and/or modify
- // it under the terms of the GNU General Public License as published by
- // the Free Software Foundation, either version 3 of the License, or
- // (at your option) any later version.
- //
- // go-ethereum is distributed in the hope that it will be useful,
- // but WITHOUT ANY WARRANTY; without even the implied warranty of
- // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- // GNU General Public License for more details.
- //
- // You should have received a copy of the GNU General Public License
- // along with go-ethereum. If not, see <http://www.gnu.org/licenses/>.
- package main
- import (
- "fmt"
- "os"
- "path/filepath"
- "sort"
- "strconv"
- "time"
- "github.com/ethereum/go-ethereum/cmd/utils"
- "github.com/ethereum/go-ethereum/common"
- "github.com/ethereum/go-ethereum/common/hexutil"
- "github.com/ethereum/go-ethereum/console/prompt"
- "github.com/ethereum/go-ethereum/core/rawdb"
- "github.com/ethereum/go-ethereum/ethdb"
- "github.com/ethereum/go-ethereum/log"
- "github.com/ethereum/go-ethereum/trie"
- "gopkg.in/urfave/cli.v1"
- )
- var (
- removedbCommand = cli.Command{
- Action: utils.MigrateFlags(removeDB),
- Name: "removedb",
- Usage: "Remove blockchain and state databases",
- ArgsUsage: "",
- Flags: []cli.Flag{
- utils.DataDirFlag,
- },
- Category: "DATABASE COMMANDS",
- Description: `
- Remove blockchain and state databases`,
- }
- dbCommand = cli.Command{
- Name: "db",
- Usage: "Low level database operations",
- ArgsUsage: "",
- Category: "DATABASE COMMANDS",
- Subcommands: []cli.Command{
- dbInspectCmd,
- dbStatCmd,
- dbCompactCmd,
- dbGetCmd,
- dbDeleteCmd,
- dbPutCmd,
- dbGetSlotsCmd,
- dbDumpFreezerIndex,
- ancientInspectCmd,
- },
- }
- dbInspectCmd = cli.Command{
- Action: utils.MigrateFlags(inspect),
- Name: "inspect",
- ArgsUsage: "<prefix> <start>",
- Flags: []cli.Flag{
- utils.DataDirFlag,
- utils.SyncModeFlag,
- utils.MainnetFlag,
- utils.RopstenFlag,
- utils.RinkebyFlag,
- utils.GoerliFlag,
- utils.YoloV3Flag,
- },
- Usage: "Inspect the storage size for each type of data in the database",
- Description: `This commands iterates the entire database. If the optional 'prefix' and 'start' arguments are provided, then the iteration is limited to the given subset of data.`,
- }
- dbStatCmd = cli.Command{
- Action: utils.MigrateFlags(dbStats),
- Name: "stats",
- Usage: "Print leveldb statistics",
- Flags: []cli.Flag{
- utils.DataDirFlag,
- utils.SyncModeFlag,
- utils.MainnetFlag,
- utils.RopstenFlag,
- utils.RinkebyFlag,
- utils.GoerliFlag,
- utils.YoloV3Flag,
- },
- }
- dbCompactCmd = cli.Command{
- Action: utils.MigrateFlags(dbCompact),
- Name: "compact",
- Usage: "Compact leveldb database. WARNING: May take a very long time",
- Flags: []cli.Flag{
- utils.DataDirFlag,
- utils.SyncModeFlag,
- utils.MainnetFlag,
- utils.RopstenFlag,
- utils.RinkebyFlag,
- utils.GoerliFlag,
- utils.YoloV3Flag,
- utils.CacheFlag,
- utils.CacheDatabaseFlag,
- },
- Description: `This command performs a database compaction.
- WARNING: This operation may take a very long time to finish, and may cause database
- corruption if it is aborted during execution'!`,
- }
- dbGetCmd = cli.Command{
- Action: utils.MigrateFlags(dbGet),
- Name: "get",
- Usage: "Show the value of a database key",
- ArgsUsage: "<hex-encoded key>",
- Flags: []cli.Flag{
- utils.DataDirFlag,
- utils.SyncModeFlag,
- utils.MainnetFlag,
- utils.RopstenFlag,
- utils.RinkebyFlag,
- utils.GoerliFlag,
- utils.YoloV3Flag,
- },
- Description: "This command looks up the specified database key from the database.",
- }
- dbDeleteCmd = cli.Command{
- Action: utils.MigrateFlags(dbDelete),
- Name: "delete",
- Usage: "Delete a database key (WARNING: may corrupt your database)",
- ArgsUsage: "<hex-encoded key>",
- Flags: []cli.Flag{
- utils.DataDirFlag,
- utils.SyncModeFlag,
- utils.MainnetFlag,
- utils.RopstenFlag,
- utils.RinkebyFlag,
- utils.GoerliFlag,
- utils.YoloV3Flag,
- },
- Description: `This command deletes the specified database key from the database.
- WARNING: This is a low-level operation which may cause database corruption!`,
- }
- dbPutCmd = cli.Command{
- Action: utils.MigrateFlags(dbPut),
- Name: "put",
- Usage: "Set the value of a database key (WARNING: may corrupt your database)",
- ArgsUsage: "<hex-encoded key> <hex-encoded value>",
- Flags: []cli.Flag{
- utils.DataDirFlag,
- utils.SyncModeFlag,
- utils.MainnetFlag,
- utils.RopstenFlag,
- utils.RinkebyFlag,
- utils.GoerliFlag,
- utils.YoloV3Flag,
- },
- Description: `This command sets a given database key to the given value.
- WARNING: This is a low-level operation which may cause database corruption!`,
- }
- dbGetSlotsCmd = cli.Command{
- Action: utils.MigrateFlags(dbDumpTrie),
- Name: "dumptrie",
- Usage: "Show the storage key/values of a given storage trie",
- ArgsUsage: "<hex-encoded storage trie root> <hex-encoded start (optional)> <int max elements (optional)>",
- Flags: []cli.Flag{
- utils.DataDirFlag,
- utils.SyncModeFlag,
- utils.MainnetFlag,
- utils.RopstenFlag,
- utils.RinkebyFlag,
- utils.GoerliFlag,
- utils.YoloV3Flag,
- },
- Description: "This command looks up the specified database key from the database.",
- }
- dbDumpFreezerIndex = cli.Command{
- Action: utils.MigrateFlags(freezerInspect),
- Name: "freezer-index",
- Usage: "Dump out the index of a given freezer type",
- ArgsUsage: "<type> <start (int)> <end (int)>",
- Flags: []cli.Flag{
- utils.DataDirFlag,
- utils.SyncModeFlag,
- utils.MainnetFlag,
- utils.RopstenFlag,
- utils.RinkebyFlag,
- utils.GoerliFlag,
- utils.YoloV3Flag,
- },
- Description: "This command displays information about the freezer index.",
- }
- ancientInspectCmd = cli.Command{
- Action: utils.MigrateFlags(ancientInspect),
- Name: "inspect-reserved-oldest-blocks",
- Flags: []cli.Flag{
- utils.DataDirFlag,
- },
- Usage: "Inspect the ancientStore information",
- Description: `This commands will read current offset from kvdb, which is the current offset and starting BlockNumber
- of ancientStore, will also displays the reserved number of blocks in ancientStore `,
- }
- )
- func removeDB(ctx *cli.Context) error {
- stack, config := makeConfigNode(ctx)
- // Remove the full node state database
- path := stack.ResolvePath("chaindata")
- if common.FileExist(path) {
- confirmAndRemoveDB(path, "full node state database")
- } else {
- log.Info("Full node state database missing", "path", path)
- }
- // Remove the full node ancient database
- path = config.Eth.DatabaseFreezer
- switch {
- case path == "":
- path = filepath.Join(stack.ResolvePath("chaindata"), "ancient")
- case !filepath.IsAbs(path):
- path = config.Node.ResolvePath(path)
- }
- if common.FileExist(path) {
- confirmAndRemoveDB(path, "full node ancient database")
- } else {
- log.Info("Full node ancient database missing", "path", path)
- }
- // Remove the light node database
- path = stack.ResolvePath("lightchaindata")
- if common.FileExist(path) {
- confirmAndRemoveDB(path, "light node database")
- } else {
- log.Info("Light node database missing", "path", path)
- }
- return nil
- }
- // confirmAndRemoveDB prompts the user for a last confirmation and removes the
- // folder if accepted.
- func confirmAndRemoveDB(database string, kind string) {
- confirm, err := prompt.Stdin.PromptConfirm(fmt.Sprintf("Remove %s (%s)?", kind, database))
- switch {
- case err != nil:
- utils.Fatalf("%v", err)
- case !confirm:
- log.Info("Database deletion skipped", "path", database)
- default:
- start := time.Now()
- filepath.Walk(database, func(path string, info os.FileInfo, err error) error {
- // If we're at the top level folder, recurse into
- if path == database {
- return nil
- }
- // Delete all the files, but not subfolders
- if !info.IsDir() {
- os.Remove(path)
- return nil
- }
- return filepath.SkipDir
- })
- log.Info("Database successfully deleted", "path", database, "elapsed", common.PrettyDuration(time.Since(start)))
- }
- }
- func inspect(ctx *cli.Context) error {
- var (
- prefix []byte
- start []byte
- )
- if ctx.NArg() > 2 {
- return fmt.Errorf("Max 2 arguments: %v", ctx.Command.ArgsUsage)
- }
- if ctx.NArg() >= 1 {
- if d, err := hexutil.Decode(ctx.Args().Get(0)); err != nil {
- return fmt.Errorf("failed to hex-decode 'prefix': %v", err)
- } else {
- prefix = d
- }
- }
- if ctx.NArg() >= 2 {
- if d, err := hexutil.Decode(ctx.Args().Get(1)); err != nil {
- return fmt.Errorf("failed to hex-decode 'start': %v", err)
- } else {
- start = d
- }
- }
- stack, _ := makeConfigNode(ctx)
- defer stack.Close()
- db := utils.MakeChainDatabase(ctx, stack, true, false)
- defer db.Close()
- return rawdb.InspectDatabase(db, prefix, start)
- }
- func ancientInspect(ctx *cli.Context) error {
- stack, _ := makeConfigNode(ctx)
- defer stack.Close()
- db := utils.MakeChainDatabase(ctx, stack, true, true)
- defer db.Close()
- return rawdb.AncientInspect(db)
- }
- func showLeveldbStats(db ethdb.Stater) {
- if stats, err := db.Stat("leveldb.stats"); err != nil {
- log.Warn("Failed to read database stats", "error", err)
- } else {
- fmt.Println(stats)
- }
- if ioStats, err := db.Stat("leveldb.iostats"); err != nil {
- log.Warn("Failed to read database iostats", "error", err)
- } else {
- fmt.Println(ioStats)
- }
- }
- func dbStats(ctx *cli.Context) error {
- stack, _ := makeConfigNode(ctx)
- defer stack.Close()
- db := utils.MakeChainDatabase(ctx, stack, true, false)
- defer db.Close()
- showLeveldbStats(db)
- return nil
- }
- func dbCompact(ctx *cli.Context) error {
- stack, _ := makeConfigNode(ctx)
- defer stack.Close()
- db := utils.MakeChainDatabase(ctx, stack, false, false)
- defer db.Close()
- log.Info("Stats before compaction")
- showLeveldbStats(db)
- log.Info("Triggering compaction")
- if err := db.Compact(nil, nil); err != nil {
- log.Info("Compact err", "error", err)
- return err
- }
- log.Info("Stats after compaction")
- showLeveldbStats(db)
- return nil
- }
- // dbGet shows the value of a given database key
- func dbGet(ctx *cli.Context) error {
- if ctx.NArg() != 1 {
- return fmt.Errorf("required arguments: %v", ctx.Command.ArgsUsage)
- }
- stack, _ := makeConfigNode(ctx)
- defer stack.Close()
- db := utils.MakeChainDatabase(ctx, stack, true, false)
- defer db.Close()
- key, err := hexutil.Decode(ctx.Args().Get(0))
- if err != nil {
- log.Info("Could not decode the key", "error", err)
- return err
- }
- data, err := db.Get(key)
- if err != nil {
- log.Info("Get operation failed", "error", err)
- return err
- }
- fmt.Printf("key %#x: %#x\n", key, data)
- return nil
- }
- // dbDelete deletes a key from the database
- func dbDelete(ctx *cli.Context) error {
- if ctx.NArg() != 1 {
- return fmt.Errorf("required arguments: %v", ctx.Command.ArgsUsage)
- }
- stack, _ := makeConfigNode(ctx)
- defer stack.Close()
- db := utils.MakeChainDatabase(ctx, stack, false, false)
- defer db.Close()
- key, err := hexutil.Decode(ctx.Args().Get(0))
- if err != nil {
- log.Info("Could not decode the key", "error", err)
- return err
- }
- data, err := db.Get(key)
- if err == nil {
- fmt.Printf("Previous value: %#x\n", data)
- }
- if err = db.Delete(key); err != nil {
- log.Info("Delete operation returned an error", "error", err)
- return err
- }
- return nil
- }
- // dbPut overwrite a value in the database
- func dbPut(ctx *cli.Context) error {
- if ctx.NArg() != 2 {
- return fmt.Errorf("required arguments: %v", ctx.Command.ArgsUsage)
- }
- stack, _ := makeConfigNode(ctx)
- defer stack.Close()
- db := utils.MakeChainDatabase(ctx, stack, false, false)
- defer db.Close()
- var (
- key []byte
- value []byte
- data []byte
- err error
- )
- key, err = hexutil.Decode(ctx.Args().Get(0))
- if err != nil {
- log.Info("Could not decode the key", "error", err)
- return err
- }
- value, err = hexutil.Decode(ctx.Args().Get(1))
- if err != nil {
- log.Info("Could not decode the value", "error", err)
- return err
- }
- data, err = db.Get(key)
- if err == nil {
- fmt.Printf("Previous value: %#x\n", data)
- }
- return db.Put(key, value)
- }
- // dbDumpTrie shows the key-value slots of a given storage trie
- func dbDumpTrie(ctx *cli.Context) error {
- if ctx.NArg() < 1 {
- return fmt.Errorf("required arguments: %v", ctx.Command.ArgsUsage)
- }
- stack, _ := makeConfigNode(ctx)
- defer stack.Close()
- db := utils.MakeChainDatabase(ctx, stack, true, false)
- defer db.Close()
- var (
- root []byte
- start []byte
- max = int64(-1)
- err error
- )
- if root, err = hexutil.Decode(ctx.Args().Get(0)); err != nil {
- log.Info("Could not decode the root", "error", err)
- return err
- }
- stRoot := common.BytesToHash(root)
- if ctx.NArg() >= 2 {
- if start, err = hexutil.Decode(ctx.Args().Get(1)); err != nil {
- log.Info("Could not decode the seek position", "error", err)
- return err
- }
- }
- if ctx.NArg() >= 3 {
- if max, err = strconv.ParseInt(ctx.Args().Get(2), 10, 64); err != nil {
- log.Info("Could not decode the max count", "error", err)
- return err
- }
- }
- theTrie, err := trie.New(stRoot, trie.NewDatabase(db))
- if err != nil {
- return err
- }
- var count int64
- it := trie.NewIterator(theTrie.NodeIterator(start))
- for it.Next() {
- if max > 0 && count == max {
- fmt.Printf("Exiting after %d values\n", count)
- break
- }
- fmt.Printf(" %d. key %#x: %#x\n", count, it.Key, it.Value)
- count++
- }
- return it.Err
- }
- func freezerInspect(ctx *cli.Context) error {
- var (
- start, end int64
- disableSnappy bool
- err error
- )
- if ctx.NArg() < 3 {
- return fmt.Errorf("required arguments: %v", ctx.Command.ArgsUsage)
- }
- kind := ctx.Args().Get(0)
- if noSnap, ok := rawdb.FreezerNoSnappy[kind]; !ok {
- var options []string
- for opt := range rawdb.FreezerNoSnappy {
- options = append(options, opt)
- }
- sort.Strings(options)
- return fmt.Errorf("Could read freezer-type '%v'. Available options: %v", kind, options)
- } else {
- disableSnappy = noSnap
- }
- if start, err = strconv.ParseInt(ctx.Args().Get(1), 10, 64); err != nil {
- log.Info("Could read start-param", "error", err)
- return err
- }
- if end, err = strconv.ParseInt(ctx.Args().Get(2), 10, 64); err != nil {
- log.Info("Could read count param", "error", err)
- return err
- }
- stack, _ := makeConfigNode(ctx)
- defer stack.Close()
- path := filepath.Join(stack.ResolvePath("chaindata"), "ancient")
- log.Info("Opening freezer", "location", path, "name", kind)
- if f, err := rawdb.NewFreezerTable(path, kind, disableSnappy); err != nil {
- return err
- } else {
- f.DumpIndex(start, end)
- }
- return nil
- }
|