github.com/neatlab/neatio@v1.7.3-0.20220425043230-d903e92fcc75/chain/neatio/chaincmd.go (about)

     1  package main
     2  
     3  import (
     4  	"fmt"
     5  	"os"
     6  	"runtime"
     7  	"strconv"
     8  	"sync/atomic"
     9  	"time"
    10  
    11  	"github.com/neatlab/neatio/neatdb"
    12  	neatptc "github.com/neatlab/neatio/neatptc"
    13  	"github.com/neatlab/neatio/params"
    14  
    15  	"github.com/neatlab/neatio/chain/core"
    16  	"github.com/neatlab/neatio/chain/core/rawdb"
    17  	"github.com/neatlab/neatio/chain/core/state"
    18  	"github.com/neatlab/neatio/chain/core/types"
    19  	"github.com/neatlab/neatio/chain/log"
    20  	"github.com/neatlab/neatio/neatptc/downloader"
    21  	"github.com/neatlab/neatio/utilities/common"
    22  	"github.com/neatlab/neatio/utilities/console"
    23  	"github.com/neatlab/neatio/utilities/event"
    24  	"github.com/neatlab/neatio/utilities/rlp"
    25  	"github.com/neatlab/neatio/utilities/utils"
    26  	"gopkg.in/urfave/cli.v1"
    27  )
    28  
    29  var (
    30  	initNEATGenesisCmd = cli.Command{
    31  		Action:    utils.MigrateFlags(initNeatGenesis),
    32  		Name:      "init-neatio",
    33  		Usage:     "Initialize NEAT genesis.json file. init-neatio {\"1000000000000000000000000000\",\"100000000000000000000000\"}",
    34  		ArgsUsage: "<genesisPath>",
    35  		Flags: []cli.Flag{
    36  			utils.DataDirFlag,
    37  		},
    38  		Category:    "BLOCKCHAIN COMMANDS",
    39  		Description: "The init-neatio initializes a new NEAT genesis.json file for the network.",
    40  	}
    41  
    42  	initCommand = cli.Command{
    43  		Action:    utils.MigrateFlags(initCmd),
    44  		Name:      "init",
    45  		Usage:     "Bootstrap and initialize a new genesis block",
    46  		ArgsUsage: "<genesisPath>",
    47  		Flags: []cli.Flag{
    48  			utils.DataDirFlag,
    49  		},
    50  		Category: "BLOCKCHAIN COMMANDS",
    51  		Description: `
    52  The init command initializes a new genesis block and definition for the network.
    53  This is a destructive action and changes the network in which you will be
    54  participating.
    55  
    56  It expects the genesis file as argument.`,
    57  	}
    58  	initSideChainCmd = cli.Command{
    59  		Action:      utils.MigrateFlags(InitSideChainCmd),
    60  		Name:        "init-side-chain",
    61  		Usage:       "neatio --sideChain=side_0,side_1,side_2 init-side-chain",
    62  		Description: "Initialize side chain genesis from chain info db",
    63  	}
    64  
    65  	createValidatorCmd = cli.Command{
    66  
    67  		Action: utils.MigrateFlags(CreatePrivateValidatorCmd),
    68  		Name:   "cvf",
    69  		Usage:  "cvf address",
    70  		Flags: []cli.Flag{
    71  			utils.DataDirFlag,
    72  		},
    73  		Description: "Create priv_validator.json for address",
    74  	}
    75  
    76  	importCommand = cli.Command{
    77  		Action:    utils.MigrateFlags(importChain),
    78  		Name:      "import",
    79  		Usage:     "Import a blockchain file",
    80  		ArgsUsage: "<chainname> <filename> (<filename 2> ... <filename N>) ",
    81  		Flags: []cli.Flag{
    82  			utils.DataDirFlag,
    83  			utils.CacheFlag,
    84  			utils.SyncModeFlag,
    85  			utils.GCModeFlag,
    86  			utils.CacheDatabaseFlag,
    87  			utils.CacheGCFlag,
    88  		},
    89  		Category: "BLOCKCHAIN COMMANDS",
    90  		Description: `
    91  The import command imports blocks from an RLP-encoded form. The form can be one file
    92  with several RLP-encoded blocks, or several files can be used.
    93  
    94  If only one file is used, import error will result in failure. If several files are used,
    95  processing will proceed even if an individual RLP-file import failure occurs.`,
    96  	}
    97  	exportCommand = cli.Command{
    98  		Action:    utils.MigrateFlags(exportChain),
    99  		Name:      "export",
   100  		Usage:     "Export blockchain into file",
   101  		ArgsUsage: "<chainname> <filename> [<blockNumFirst> <blockNumLast>]",
   102  		Flags: []cli.Flag{
   103  			utils.DataDirFlag,
   104  			utils.CacheFlag,
   105  			utils.SyncModeFlag,
   106  		},
   107  		Category: "BLOCKCHAIN COMMANDS",
   108  		Description: `
   109  Requires a first argument of the file to write to.
   110  Optional second and third arguments control the first and
   111  last block to write. In this mode, the file will be appended
   112  if already existing.`,
   113  	}
   114  	importPreimagesCommand = cli.Command{
   115  		Action:    utils.MigrateFlags(importPreimages),
   116  		Name:      "import-preimages",
   117  		Usage:     "Import the preimage database from an RLP stream",
   118  		ArgsUsage: "<datafile>",
   119  		Flags: []cli.Flag{
   120  			utils.DataDirFlag,
   121  			utils.CacheFlag,
   122  			utils.SyncModeFlag,
   123  		},
   124  		Category: "BLOCKCHAIN COMMANDS",
   125  		Description: `
   126  	The import-preimages command imports hash preimages from an RLP encoded stream.`,
   127  	}
   128  	exportPreimagesCommand = cli.Command{
   129  		Action:    utils.MigrateFlags(exportPreimages),
   130  		Name:      "export-preimages",
   131  		Usage:     "Export the preimage database into an RLP stream",
   132  		ArgsUsage: "<dumpfile>",
   133  		Flags: []cli.Flag{
   134  			utils.DataDirFlag,
   135  			utils.CacheFlag,
   136  			utils.SyncModeFlag,
   137  		},
   138  		Category: "BLOCKCHAIN COMMANDS",
   139  		Description: `
   140  The export-preimages command export hash preimages to an RLP encoded stream`,
   141  	}
   142  	copydbCommand = cli.Command{
   143  		Action:    utils.MigrateFlags(copyDb),
   144  		Name:      "copydb",
   145  		Usage:     "Create a local chain from a target chaindata folder",
   146  		ArgsUsage: "<sourceChaindataDir>",
   147  		Flags: []cli.Flag{
   148  			utils.DataDirFlag,
   149  			utils.CacheFlag,
   150  			utils.SyncModeFlag,
   151  			utils.TestnetFlag,
   152  		},
   153  		Category: "BLOCKCHAIN COMMANDS",
   154  		Description: `
   155  The first argument must be the directory containing the blockchain to download from`,
   156  	}
   157  	removedbCommand = cli.Command{
   158  		Action:    utils.MigrateFlags(removeDB),
   159  		Name:      "removedb",
   160  		Usage:     "Remove blockchain and state databases",
   161  		ArgsUsage: " ",
   162  		Flags: []cli.Flag{
   163  			utils.DataDirFlag,
   164  		},
   165  		Category: "BLOCKCHAIN COMMANDS",
   166  		Description: `
   167  Remove blockchain and state databases`,
   168  	}
   169  	dumpCommand = cli.Command{
   170  		Action:    utils.MigrateFlags(dump),
   171  		Name:      "dump",
   172  		Usage:     "Dump a specific block from storage",
   173  		ArgsUsage: "[<blockHash> | <blockNum>]...",
   174  		Flags: []cli.Flag{
   175  			utils.DataDirFlag,
   176  			utils.CacheFlag,
   177  		},
   178  		Category: "BLOCKCHAIN COMMANDS",
   179  		Description: `
   180  The arguments are interpreted as block numbers or hashes.
   181  Use "ethereum dump 0" to dump the genesis block.`,
   182  	}
   183  	countBlockStateCommand = cli.Command{
   184  		Action:    utils.MigrateFlags(countBlockState),
   185  		Name:      "count-blockstate",
   186  		Usage:     "Count the block state",
   187  		ArgsUsage: "<datafile>",
   188  		Flags: []cli.Flag{
   189  			utils.DataDirFlag,
   190  			utils.CacheFlag,
   191  			utils.SyncModeFlag,
   192  		},
   193  		Category: "BLOCKCHAIN COMMANDS",
   194  		Description: `
   195  	The count-blockstate command count the block state from a given height.`,
   196  	}
   197  
   198  	versionCommand = cli.Command{
   199  		Action:    utils.MigrateFlags(version),
   200  		Name:      "version",
   201  		Usage:     "Print version numbers",
   202  		ArgsUsage: " ",
   203  		Category:  "MISCELLANEOUS COMMANDS",
   204  		Description: `
   205  The output of this command is supposed to be machine-readable.
   206  `,
   207  	}
   208  )
   209  
   210  func importChain(ctx *cli.Context) error {
   211  	if len(ctx.Args()) < 1 {
   212  		utils.Fatalf("This command requires an argument.")
   213  	}
   214  
   215  	chainName := ctx.Args().First()
   216  	if chainName == "" {
   217  		utils.Fatalf("This command requires chain name specified.")
   218  	}
   219  
   220  	stack, cfg := makeConfigNode(ctx, chainName)
   221  	cch := GetCMInstance(ctx).cch
   222  	utils.RegisterIntService(stack, &cfg.Eth, ctx, cch)
   223  
   224  	defer stack.Close()
   225  
   226  	chain, db := utils.MakeChain(ctx, stack)
   227  	defer db.Close()
   228  
   229  	var peakMemAlloc, peakMemSys uint64
   230  	go func() {
   231  		stats := new(runtime.MemStats)
   232  		for {
   233  			runtime.ReadMemStats(stats)
   234  			if atomic.LoadUint64(&peakMemAlloc) < stats.Alloc {
   235  				atomic.StoreUint64(&peakMemAlloc, stats.Alloc)
   236  			}
   237  			if atomic.LoadUint64(&peakMemSys) < stats.Sys {
   238  				atomic.StoreUint64(&peakMemSys, stats.Sys)
   239  			}
   240  			time.Sleep(5 * time.Second)
   241  		}
   242  	}()
   243  
   244  	start := time.Now()
   245  
   246  	if len(ctx.Args()) == 2 {
   247  		if err := utils.ImportChain(chain, ctx.Args().Get(1)); err != nil {
   248  			log.Error("Import error", "err", err)
   249  		}
   250  	} else {
   251  		for i, arg := range ctx.Args() {
   252  			if i == 0 {
   253  				continue
   254  			}
   255  			if err := utils.ImportChain(chain, arg); err != nil {
   256  				log.Error("Import error", "file", arg, "err", err)
   257  			}
   258  		}
   259  	}
   260  	chain.Stop()
   261  	fmt.Printf("Import done in %v.\n\n", time.Since(start))
   262  
   263  	stats, err := db.Stat("leveldb.stats")
   264  	if err != nil {
   265  		utils.Fatalf("Failed to read database stats: %v", err)
   266  	}
   267  	fmt.Println(stats)
   268  
   269  	ioStats, err := db.Stat("leveldb.iostats")
   270  	if err != nil {
   271  		utils.Fatalf("Failed to read database iostats: %v", err)
   272  	}
   273  	fmt.Println(ioStats)
   274  
   275  	mem := new(runtime.MemStats)
   276  	runtime.ReadMemStats(mem)
   277  
   278  	fmt.Printf("Object memory: %.3f MB current, %.3f MB peak\n", float64(mem.Alloc)/1024/1024, float64(atomic.LoadUint64(&peakMemAlloc))/1024/1024)
   279  	fmt.Printf("System memory: %.3f MB current, %.3f MB peak\n", float64(mem.Sys)/1024/1024, float64(atomic.LoadUint64(&peakMemSys))/1024/1024)
   280  	fmt.Printf("Allocations:   %.3f million\n", float64(mem.Mallocs)/1000000)
   281  	fmt.Printf("GC pause:      %v\n\n", time.Duration(mem.PauseTotalNs))
   282  
   283  	if ctx.GlobalIsSet(utils.NoCompactionFlag.Name) {
   284  		return nil
   285  	}
   286  
   287  	start = time.Now()
   288  	fmt.Println("Compacting entire database...")
   289  	if err = db.Compact(nil, nil); err != nil {
   290  		utils.Fatalf("Compaction failed: %v", err)
   291  	}
   292  	fmt.Printf("Compaction done in %v.\n\n", time.Since(start))
   293  
   294  	stats, err = db.Stat("leveldb.stats")
   295  	if err != nil {
   296  		utils.Fatalf("Failed to read database stats: %v", err)
   297  	}
   298  	fmt.Println(stats)
   299  
   300  	ioStats, err = db.Stat("leveldb.iostats")
   301  	if err != nil {
   302  		utils.Fatalf("Failed to read database iostats: %v", err)
   303  	}
   304  	fmt.Println(ioStats)
   305  
   306  	return nil
   307  }
   308  
   309  func exportChain(ctx *cli.Context) error {
   310  	if len(ctx.Args()) < 1 {
   311  		utils.Fatalf("This command requires an argument.")
   312  	}
   313  
   314  	chainName := ctx.Args().First()
   315  	if chainName == "" {
   316  		utils.Fatalf("This command requires chain name specified.")
   317  	}
   318  
   319  	stack, cfg := makeConfigNode(ctx, chainName)
   320  	utils.RegisterIntService(stack, &cfg.Eth, ctx, GetCMInstance(ctx).cch)
   321  
   322  	defer stack.Close()
   323  
   324  	chain, _ := utils.MakeChain(ctx, stack)
   325  	start := time.Now()
   326  
   327  	var err error
   328  	fp := ctx.Args().Get(1)
   329  	if len(ctx.Args()) < 4 {
   330  		err = utils.ExportChain(chain, fp)
   331  	} else {
   332  
   333  		first, ferr := strconv.ParseInt(ctx.Args().Get(2), 10, 64)
   334  		last, lerr := strconv.ParseInt(ctx.Args().Get(3), 10, 64)
   335  		if ferr != nil || lerr != nil {
   336  			utils.Fatalf("Export error in parsing parameters: block number not an integer\n")
   337  		}
   338  		if first < 0 || last < 0 {
   339  			utils.Fatalf("Export error: block number must be greater than 0\n")
   340  		}
   341  		err = utils.ExportAppendChain(chain, fp, uint64(first), uint64(last))
   342  	}
   343  
   344  	if err != nil {
   345  		utils.Fatalf("Export error: %v\n", err)
   346  	}
   347  	fmt.Printf("Export done in %v", time.Since(start))
   348  	return nil
   349  }
   350  
   351  func importPreimages(ctx *cli.Context) error {
   352  	if len(ctx.Args()) < 1 {
   353  		utils.Fatalf("This command requires an argument.")
   354  	}
   355  
   356  	chainName := ctx.Args().Get(1)
   357  	if chainName == "" {
   358  		utils.Fatalf("This command requires chain name specified.")
   359  	}
   360  
   361  	stack, cfg := makeConfigNode(ctx, chainName)
   362  	utils.RegisterIntService(stack, &cfg.Eth, ctx, GetCMInstance(ctx).cch)
   363  	defer stack.Close()
   364  
   365  	db := utils.MakeChainDatabase(ctx, stack)
   366  	start := time.Now()
   367  
   368  	if err := utils.ImportPreimages(db, ctx.Args().First()); err != nil {
   369  		utils.Fatalf("Import error: %v\n", err)
   370  	}
   371  	fmt.Printf("Import done in %v\n", time.Since(start))
   372  	return nil
   373  }
   374  
   375  func exportPreimages(ctx *cli.Context) error {
   376  	if len(ctx.Args()) < 1 {
   377  		utils.Fatalf("This command requires an argument.")
   378  	}
   379  
   380  	chainName := ctx.Args().Get(1)
   381  	if chainName == "" {
   382  		utils.Fatalf("This command requires chain name specified.")
   383  	}
   384  
   385  	stack, cfg := makeConfigNode(ctx, chainName)
   386  	utils.RegisterIntService(stack, &cfg.Eth, ctx, GetCMInstance(ctx).cch)
   387  	defer stack.Close()
   388  
   389  	db := utils.MakeChainDatabase(ctx, stack)
   390  	start := time.Now()
   391  
   392  	if err := utils.ExportPreimages(db, ctx.Args().First()); err != nil {
   393  		utils.Fatalf("Export error: %v\n", err)
   394  	}
   395  	fmt.Printf("Export done in %v\n", time.Since(start))
   396  	return nil
   397  }
   398  func copyDb(ctx *cli.Context) error {
   399  
   400  	if len(ctx.Args()) != 1 {
   401  		utils.Fatalf("Source chaindata directory path argument missing")
   402  	}
   403  
   404  	chainName := ctx.Args().Get(1)
   405  	if chainName == "" {
   406  		utils.Fatalf("This command requires chain name specified.")
   407  	}
   408  
   409  	stack, _ := makeConfigNode(ctx, chainName)
   410  	chain, chainDb := utils.MakeChain(ctx, stack)
   411  
   412  	syncmode := *utils.GlobalTextMarshaler(ctx, utils.SyncModeFlag.Name).(*downloader.SyncMode)
   413  	dl := downloader.New(syncmode, chainDb, new(event.TypeMux), chain, nil, nil, nil)
   414  
   415  	db, err := rawdb.NewLevelDBDatabase(ctx.Args().First(), ctx.GlobalInt(utils.CacheFlag.Name), 256, "")
   416  	if err != nil {
   417  		return err
   418  	}
   419  	hc, err := core.NewHeaderChain(db, chain.Config(), chain.Engine(), func() bool { return false })
   420  	if err != nil {
   421  		return err
   422  	}
   423  	peer := downloader.NewFakePeer("local", db, hc, dl)
   424  	if err = dl.RegisterPeer("local", 63, peer); err != nil {
   425  		return err
   426  	}
   427  
   428  	start := time.Now()
   429  
   430  	currentHeader := hc.CurrentHeader()
   431  	if err = dl.Synchronise("local", currentHeader.Hash(), hc.GetTd(currentHeader.Hash(), currentHeader.Number.Uint64()), syncmode); err != nil {
   432  		return err
   433  	}
   434  	for dl.Synchronising() {
   435  		time.Sleep(10 * time.Millisecond)
   436  	}
   437  	fmt.Printf("Database copy done in %v\n", time.Since(start))
   438  
   439  	start = time.Now()
   440  	fmt.Println("Compacting entire database...")
   441  	if err = db.Compact(nil, nil); err != nil {
   442  		utils.Fatalf("Compaction failed: %v", err)
   443  	}
   444  	fmt.Printf("Compaction done in %v.\n\n", time.Since(start))
   445  
   446  	return nil
   447  }
   448  
   449  func removeDB(ctx *cli.Context) error {
   450  	chainName := ctx.Args().Get(1)
   451  	if chainName == "" {
   452  		utils.Fatalf("This command requires chain name specified.")
   453  	}
   454  
   455  	stack, _ := makeConfigNode(ctx, chainName)
   456  
   457  	for _, name := range []string{"chaindata", "lightchaindata"} {
   458  
   459  		logger := log.New("database", name)
   460  
   461  		dbdir := stack.ResolvePath(name)
   462  		if !common.FileExist(dbdir) {
   463  			logger.Info("Database doesn't exist, skipping", "path", dbdir)
   464  			continue
   465  		}
   466  
   467  		fmt.Println(dbdir)
   468  		confirm, err := console.Stdin.PromptConfirm("Remove this database?")
   469  		switch {
   470  		case err != nil:
   471  			utils.Fatalf("%v", err)
   472  		case !confirm:
   473  			logger.Warn("Database deletion aborted")
   474  		default:
   475  			start := time.Now()
   476  			os.RemoveAll(dbdir)
   477  			logger.Info("Database successfully deleted", "elapsed", common.PrettyDuration(time.Since(start)))
   478  		}
   479  	}
   480  	return nil
   481  }
   482  
   483  func dump(ctx *cli.Context) error {
   484  	chainName := ctx.Args().Get(1)
   485  	if chainName == "" {
   486  		utils.Fatalf("This command requires chain name specified.")
   487  	}
   488  
   489  	stack, _ := makeConfigNode(ctx, chainName)
   490  	chain, chainDb := utils.MakeChain(ctx, stack)
   491  	for _, arg := range ctx.Args() {
   492  		var block *types.Block
   493  		if hashish(arg) {
   494  			block = chain.GetBlockByHash(common.HexToHash(arg))
   495  		} else {
   496  			num, _ := strconv.Atoi(arg)
   497  			block = chain.GetBlockByNumber(uint64(num))
   498  		}
   499  		if block == nil {
   500  			fmt.Println("{}")
   501  			utils.Fatalf("block not found")
   502  		} else {
   503  			state, err := state.New(block.Root(), state.NewDatabase(chainDb))
   504  			if err != nil {
   505  				utils.Fatalf("could not create new state: %v", err)
   506  			}
   507  			fmt.Printf("%s\n", state.Dump())
   508  		}
   509  	}
   510  	chainDb.Close()
   511  	return nil
   512  }
   513  
   514  func hashish(x string) bool {
   515  	_, err := strconv.Atoi(x)
   516  	return err != nil
   517  }
   518  
   519  func countBlockState(ctx *cli.Context) error {
   520  	if len(ctx.Args()) < 1 {
   521  		utils.Fatalf("This command requires an argument.")
   522  	}
   523  
   524  	chainName := ctx.Args().Get(1)
   525  	if chainName == "" {
   526  		utils.Fatalf("This command requires chain name specified.")
   527  	}
   528  
   529  	stack, cfg := makeConfigNode(ctx, chainName)
   530  	utils.RegisterIntService(stack, &cfg.Eth, ctx, GetCMInstance(ctx).cch)
   531  	defer stack.Close()
   532  
   533  	chainDb := utils.MakeChainDatabase(ctx, stack)
   534  
   535  	height, _ := strconv.ParseUint(ctx.Args().First(), 10, 64)
   536  
   537  	blockhash := rawdb.ReadCanonicalHash(chainDb, height)
   538  	block := rawdb.ReadBlock(chainDb, blockhash, height)
   539  	bsize := block.Size()
   540  
   541  	root := block.Header().Root
   542  	statedb, _ := state.New(block.Root(), state.NewDatabase(chainDb))
   543  	accountTrie, _ := statedb.Database().OpenTrie(root)
   544  
   545  	count := CountSize{}
   546  	countTrie(chainDb, accountTrie, &count, func(addr common.Address, account state.Account) {
   547  		if account.Root != emptyRoot {
   548  			storageTrie, _ := statedb.Database().OpenStorageTrie(common.Hash{}, account.Root)
   549  			countTrie(chainDb, storageTrie, &count, nil)
   550  		}
   551  
   552  		if account.TX1Root != emptyRoot {
   553  			tx1Trie, _ := statedb.Database().OpenTX1Trie(common.Hash{}, account.TX1Root)
   554  			countTrie(chainDb, tx1Trie, &count, nil)
   555  		}
   556  
   557  		if account.TX3Root != emptyRoot {
   558  			tx3Trie, _ := statedb.Database().OpenTX3Trie(common.Hash{}, account.TX3Root)
   559  			countTrie(chainDb, tx3Trie, &count, nil)
   560  		}
   561  
   562  		if account.ProxiedRoot != emptyRoot {
   563  			proxiedTrie, _ := statedb.Database().OpenProxiedTrie(common.Hash{}, account.ProxiedRoot)
   564  			countTrie(chainDb, proxiedTrie, &count, nil)
   565  		}
   566  
   567  		if account.RewardRoot != emptyRoot {
   568  			rewardTrie, _ := statedb.Database().OpenRewardTrie(common.Hash{}, account.RewardRoot)
   569  			countTrie(chainDb, rewardTrie, &count, nil)
   570  		}
   571  	})
   572  
   573  	fh, err := os.OpenFile("blockstate_nodedump", os.O_CREATE|os.O_WRONLY|os.O_TRUNC, os.ModePerm)
   574  	if err != nil {
   575  		return err
   576  	}
   577  	defer fh.Close()
   578  
   579  	for _, data := range count.Data {
   580  		fh.WriteString(data.key + " " + data.value + "\n")
   581  	}
   582  
   583  	fmt.Printf("Block %d, block size %v, state node %v, state size %v\n", height, bsize, count.Totalnode, count.Totalnodevaluesize)
   584  	return nil
   585  }
   586  
   587  type CountSize struct {
   588  	Totalnodevaluesize, Totalnode int
   589  	Data                          []nodeData
   590  }
   591  
   592  type nodeData struct {
   593  	key, value string
   594  }
   595  
   596  type processLeafTrie func(addr common.Address, account state.Account)
   597  
   598  var emptyRoot = common.HexToHash("56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421")
   599  
   600  func countTrie(db neatdb.Database, t state.Trie, count *CountSize, processLeaf processLeafTrie) {
   601  	for it := t.NodeIterator(nil); it.Next(true); {
   602  		if !it.Leaf() {
   603  
   604  			node, _ := db.Get(it.Hash().Bytes())
   605  			count.Totalnodevaluesize += len(node)
   606  			count.Totalnode++
   607  			count.Data = append(count.Data, nodeData{it.Hash().String(), common.Bytes2Hex(node)})
   608  		} else {
   609  
   610  			if processLeaf != nil {
   611  				addr := t.GetKey(it.LeafKey())
   612  				if len(addr) == 20 {
   613  					var data state.Account
   614  					rlp.DecodeBytes(it.LeafBlob(), &data)
   615  
   616  					processLeaf(common.BytesToAddress(addr), data)
   617  				}
   618  			}
   619  		}
   620  	}
   621  }
   622  
   623  func version(ctx *cli.Context) error {
   624  	fmt.Println("Chain:", clientIdentifier)
   625  	fmt.Println("Version:", params.VersionWithMeta)
   626  	if gitCommit != "" {
   627  		fmt.Println("Git Commit:", gitCommit)
   628  	}
   629  	if gitDate != "" {
   630  		fmt.Println("Git Commit Date:", gitDate)
   631  	}
   632  	fmt.Println("Architecture:", runtime.GOARCH)
   633  	fmt.Println("Protocol Versions:", neatptc.ProtocolVersions)
   634  	fmt.Println("Go Version:", runtime.Version())
   635  	fmt.Println("Operating System:", runtime.GOOS)
   636  	fmt.Printf("GOPATH=%s\n", os.Getenv("GOPATH"))
   637  	fmt.Printf("GOROOT=%s\n", runtime.GOROOT())
   638  	return nil
   639  }