| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684 |
- /*
- 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/>.
- */
- /**
- * @authors
- * Jeffrey Wilcke <i@jev.io>
- */
- package main
- import (
- "bufio"
- "fmt"
- "io"
- "io/ioutil"
- "os"
- "path/filepath"
- "runtime"
- "strconv"
- "strings"
- "time"
- "github.com/codegangsta/cli"
- "github.com/ethereum/ethash"
- "github.com/ethereum/go-ethereum/accounts"
- "github.com/ethereum/go-ethereum/cmd/utils"
- "github.com/ethereum/go-ethereum/common"
- "github.com/ethereum/go-ethereum/core"
- "github.com/ethereum/go-ethereum/core/state"
- "github.com/ethereum/go-ethereum/core/types"
- "github.com/ethereum/go-ethereum/eth"
- "github.com/ethereum/go-ethereum/logger"
- "github.com/mattn/go-colorable"
- "github.com/mattn/go-isatty"
- "github.com/peterh/liner"
- )
- import _ "net/http/pprof"
- const (
- ClientIdentifier = "Geth"
- Version = "0.9.19"
- )
- var (
- gitCommit string // set via linker flag
- nodeNameVersion string
- app *cli.App
- )
- func init() {
- if gitCommit == "" {
- nodeNameVersion = Version
- } else {
- nodeNameVersion = Version + "-" + gitCommit[:8]
- }
- app = utils.NewApp(Version, "the go-ethereum command line interface")
- app.Action = run
- app.HideVersion = true // we have a command to print the version
- app.Commands = []cli.Command{
- blocktestCmd,
- {
- Action: makedag,
- Name: "makedag",
- Usage: "generate ethash dag (for testing)",
- Description: `
- The makedag command generates an ethash DAG in /tmp/dag.
- This command exists to support the system testing project.
- Regular users do not need to execute it.
- `,
- },
- {
- Action: version,
- Name: "version",
- Usage: "print ethereum version numbers",
- Description: `
- The output of this command is supposed to be machine-readable.
- `,
- },
- {
- Name: "wallet",
- Usage: "ethereum presale wallet",
- Subcommands: []cli.Command{
- {
- Action: importWallet,
- Name: "import",
- Usage: "import ethereum presale wallet",
- },
- },
- },
- {
- Action: accountList,
- Name: "account",
- Usage: "manage accounts",
- Description: `
- Manage accounts lets you create new accounts, list all existing accounts,
- import a private key into a new account.
- 'account help' shows a list of subcommands or help for one subcommand.
- It supports interactive mode, when you are prompted for password as well as
- non-interactive mode where passwords are supplied via a given password file.
- Non-interactive mode is only meant for scripted use on test networks or known
- safe environments.
- Make sure you remember the password you gave when creating a new account (with
- either new or import). Without it you are not able to unlock your account.
- Note that exporting your key in unencrypted format is NOT supported.
- Keys are stored under <DATADIR>/keys.
- It is safe to transfer the entire directory or the individual keys therein
- between ethereum nodes.
- Make sure you backup your keys regularly.
- And finally. DO NOT FORGET YOUR PASSWORD.
- `,
- Subcommands: []cli.Command{
- {
- Action: accountList,
- Name: "list",
- Usage: "print account addresses",
- },
- {
- Action: accountCreate,
- Name: "new",
- Usage: "create a new account",
- Description: `
- ethereum account new
- Creates a new account. Prints the address.
- The account is saved in encrypted format, you are prompted for a passphrase.
- You must remember this passphrase to unlock your account in the future.
- For non-interactive use the passphrase can be specified with the --password flag:
- ethereum --password <passwordfile> account new
- Note, this is meant to be used for testing only, it is a bad idea to save your
- password to file or expose in any other way.
- `,
- },
- {
- Action: accountImport,
- Name: "import",
- Usage: "import a private key into a new account",
- Description: `
- ethereum account import <keyfile>
- Imports an unencrypted private key from <keyfile> and creates a new account.
- Prints the address.
- The keyfile is assumed to contain an unencrypted private key in hexadecimal format.
- The account is saved in encrypted format, you are prompted for a passphrase.
- You must remember this passphrase to unlock your account in the future.
- For non-interactive use the passphrase can be specified with the -password flag:
- ethereum --password <passwordfile> account import <keyfile>
- Note:
- As you can directly copy your encrypted accounts to another ethereum instance,
- this import mechanism is not needed when you transfer an account between
- nodes.
- `,
- },
- },
- },
- {
- Action: dump,
- Name: "dump",
- Usage: `dump a specific block from storage`,
- Description: `
- The arguments are interpreted as block numbers or hashes.
- Use "ethereum dump 0" to dump the genesis block.
- `,
- },
- {
- Action: console,
- Name: "console",
- Usage: `Geth Console: interactive JavaScript environment`,
- Description: `
- The Geth console is an interactive shell for the JavaScript runtime environment
- which exposes a node admin interface as well as the Ðapp JavaScript API.
- See https://github.com/ethereum/go-ethereum/wiki/Javascipt-Console
- `,
- },
- {
- Action: execJSFiles,
- Name: "js",
- Usage: `executes the given JavaScript files in the Geth JavaScript VM`,
- Description: `
- The JavaScript VM exposes a node admin interface as well as the Ðapp
- JavaScript API. See https://github.com/ethereum/go-ethereum/wiki/Javascipt-Console
- `,
- },
- {
- Action: importchain,
- Name: "import",
- Usage: `import a blockchain file`,
- },
- {
- Action: exportchain,
- Name: "export",
- Usage: `export blockchain into file`,
- },
- {
- Action: upgradeDb,
- Name: "upgradedb",
- Usage: "upgrade chainblock database",
- },
- }
- app.Flags = []cli.Flag{
- utils.IdentityFlag,
- utils.UnlockedAccountFlag,
- utils.PasswordFileFlag,
- utils.BootnodesFlag,
- utils.DataDirFlag,
- utils.BlockchainVersionFlag,
- utils.JSpathFlag,
- utils.ListenPortFlag,
- utils.MaxPeersFlag,
- utils.MaxPendingPeersFlag,
- utils.EtherbaseFlag,
- utils.GasPriceFlag,
- utils.MinerThreadsFlag,
- utils.MiningEnabledFlag,
- utils.NATFlag,
- utils.NatspecEnabledFlag,
- utils.NodeKeyFileFlag,
- utils.NodeKeyHexFlag,
- utils.RPCEnabledFlag,
- utils.RPCListenAddrFlag,
- utils.RPCPortFlag,
- utils.WhisperEnabledFlag,
- utils.VMDebugFlag,
- utils.ProtocolVersionFlag,
- utils.NetworkIdFlag,
- utils.RPCCORSDomainFlag,
- utils.VerbosityFlag,
- utils.BacktraceAtFlag,
- utils.LogToStdErrFlag,
- utils.LogVModuleFlag,
- utils.LogFileFlag,
- utils.LogJSONFlag,
- utils.PProfEanbledFlag,
- utils.PProfPortFlag,
- utils.SolcPathFlag,
- }
- app.Before = func(ctx *cli.Context) error {
- if ctx.GlobalBool(utils.PProfEanbledFlag.Name) {
- utils.StartPProf(ctx)
- }
- return nil
- }
- // missing:
- // flag.StringVar(&ConfigFile, "conf", defaultConfigFile, "config file")
- // flag.BoolVar(&DiffTool, "difftool", false, "creates output for diff'ing. Sets LogLevel=0")
- // flag.StringVar(&DiffType, "diff", "all", "sets the level of diff output [vm, all]. Has no effect if difftool=false")
- }
- func main() {
- //fmt.Printf("\n 🌞\n\n ᴡᴇʟᴄᴏᴍᴇ ᴛᴏ ᴛʜᴇ\n 𝐅 𝐑 𝐎 𝐍 𝐓 𝐈 𝐄 𝐑\n\n🌾 🌵🌾🌾 🐎 🌾 🌵 🌾\n\n")
- fmt.Println("\n Welcome to the\n FRONTIER\n")
- runtime.GOMAXPROCS(runtime.NumCPU())
- defer logger.Flush()
- if err := app.Run(os.Args); err != nil {
- fmt.Fprintln(os.Stderr, err)
- os.Exit(1)
- }
- }
- func run(ctx *cli.Context) {
- utils.HandleInterrupt()
- cfg := utils.MakeEthConfig(ClientIdentifier, nodeNameVersion, ctx)
- ethereum, err := eth.New(cfg)
- if err != nil {
- utils.Fatalf("%v", err)
- }
- startEth(ctx, ethereum)
- // this blocks the thread
- ethereum.WaitForShutdown()
- }
- func console(ctx *cli.Context) {
- // Wrap the standard output with a colorified stream (windows)
- if isatty.IsTerminal(os.Stdout.Fd()) {
- if pr, pw, err := os.Pipe(); err == nil {
- go io.Copy(colorable.NewColorableStdout(), pr)
- os.Stdout = pw
- }
- }
- cfg := utils.MakeEthConfig(ClientIdentifier, nodeNameVersion, ctx)
- ethereum, err := eth.New(cfg)
- if err != nil {
- utils.Fatalf("%v", err)
- }
- startEth(ctx, ethereum)
- repl := newJSRE(
- ethereum,
- ctx.String(utils.JSpathFlag.Name),
- ctx.String(utils.SolcPathFlag.Name),
- ctx.GlobalString(utils.RPCCORSDomainFlag.Name),
- true,
- nil,
- )
- repl.interactive()
- ethereum.Stop()
- ethereum.WaitForShutdown()
- }
- func execJSFiles(ctx *cli.Context) {
- cfg := utils.MakeEthConfig(ClientIdentifier, nodeNameVersion, ctx)
- ethereum, err := eth.New(cfg)
- if err != nil {
- utils.Fatalf("%v", err)
- }
- startEth(ctx, ethereum)
- repl := newJSRE(
- ethereum,
- ctx.String(utils.JSpathFlag.Name),
- ctx.String(utils.SolcPathFlag.Name),
- ctx.GlobalString(utils.RPCCORSDomainFlag.Name),
- false,
- nil,
- )
- for _, file := range ctx.Args() {
- repl.exec(file)
- }
- ethereum.Stop()
- ethereum.WaitForShutdown()
- }
- func unlockAccount(ctx *cli.Context, am *accounts.Manager, account string) (passphrase string) {
- var err error
- // Load startup keys. XXX we are going to need a different format
- // Attempt to unlock the account
- passphrase = getPassPhrase(ctx, "", false)
- accbytes := common.FromHex(account)
- if len(accbytes) == 0 {
- utils.Fatalf("Invalid account address '%s'", account)
- }
- err = am.Unlock(accbytes, passphrase)
- if err != nil {
- utils.Fatalf("Unlock account failed '%v'", err)
- }
- return
- }
- func startEth(ctx *cli.Context, eth *eth.Ethereum) {
- // Start Ethereum itself
- utils.StartEthereum(eth)
- am := eth.AccountManager()
- account := ctx.GlobalString(utils.UnlockedAccountFlag.Name)
- if len(account) > 0 {
- if account == "primary" {
- accbytes, err := am.Primary()
- if err != nil {
- utils.Fatalf("no primary account: %v", err)
- }
- account = common.ToHex(accbytes)
- }
- unlockAccount(ctx, am, account)
- }
- // Start auxiliary services if enabled.
- if ctx.GlobalBool(utils.RPCEnabledFlag.Name) {
- if err := utils.StartRPC(eth, ctx); err != nil {
- utils.Fatalf("Error starting RPC: %v", err)
- }
- }
- if ctx.GlobalBool(utils.MiningEnabledFlag.Name) {
- if err := eth.StartMining(ctx.GlobalInt(utils.MinerThreadsFlag.Name)); err != nil {
- utils.Fatalf("%v", err)
- }
- }
- }
- func accountList(ctx *cli.Context) {
- am := utils.GetAccountManager(ctx)
- accts, err := am.Accounts()
- if err != nil {
- utils.Fatalf("Could not list accounts: %v", err)
- }
- name := "Primary"
- for i, acct := range accts {
- fmt.Printf("%s #%d: %x\n", name, i, acct)
- name = "Account"
- }
- }
- func getPassPhrase(ctx *cli.Context, desc string, confirmation bool) (passphrase string) {
- passfile := ctx.GlobalString(utils.PasswordFileFlag.Name)
- if len(passfile) == 0 {
- fmt.Println(desc)
- auth, err := readPassword("Passphrase: ", true)
- if err != nil {
- utils.Fatalf("%v", err)
- }
- if confirmation {
- confirm, err := readPassword("Repeat Passphrase: ", false)
- if err != nil {
- utils.Fatalf("%v", err)
- }
- if auth != confirm {
- utils.Fatalf("Passphrases did not match.")
- }
- }
- passphrase = auth
- } else {
- passbytes, err := ioutil.ReadFile(passfile)
- if err != nil {
- utils.Fatalf("Unable to read password file '%s': %v", passfile, err)
- }
- passphrase = string(passbytes)
- }
- return
- }
- func accountCreate(ctx *cli.Context) {
- am := utils.GetAccountManager(ctx)
- passphrase := getPassPhrase(ctx, "Your new account is locked with a password. Please give a password. Do not forget this password.", true)
- acct, err := am.NewAccount(passphrase)
- if err != nil {
- utils.Fatalf("Could not create the account: %v", err)
- }
- fmt.Printf("Address: %x\n", acct)
- }
- func importWallet(ctx *cli.Context) {
- keyfile := ctx.Args().First()
- if len(keyfile) == 0 {
- utils.Fatalf("keyfile must be given as argument")
- }
- keyJson, err := ioutil.ReadFile(keyfile)
- if err != nil {
- utils.Fatalf("Could not read wallet file: %v", err)
- }
- am := utils.GetAccountManager(ctx)
- passphrase := getPassPhrase(ctx, "", false)
- acct, err := am.ImportPreSaleKey(keyJson, passphrase)
- if err != nil {
- utils.Fatalf("Could not create the account: %v", err)
- }
- fmt.Printf("Address: %x\n", acct)
- }
- func accountImport(ctx *cli.Context) {
- keyfile := ctx.Args().First()
- if len(keyfile) == 0 {
- utils.Fatalf("keyfile must be given as argument")
- }
- am := utils.GetAccountManager(ctx)
- passphrase := getPassPhrase(ctx, "Your new account is locked with a password. Please give a password. Do not forget this password.", true)
- acct, err := am.Import(keyfile, passphrase)
- if err != nil {
- utils.Fatalf("Could not create the account: %v", err)
- }
- fmt.Printf("Address: %x\n", acct)
- }
- func importchain(ctx *cli.Context) {
- if len(ctx.Args()) != 1 {
- utils.Fatalf("This command requires an argument.")
- }
- cfg := utils.MakeEthConfig(ClientIdentifier, Version, ctx)
- cfg.SkipBcVersionCheck = true
- ethereum, err := eth.New(cfg)
- if err != nil {
- utils.Fatalf("%v\n", err)
- }
- chainmgr := ethereum.ChainManager()
- start := time.Now()
- err = utils.ImportChain(chainmgr, ctx.Args().First())
- if err != nil {
- utils.Fatalf("Import error: %v\n", err)
- }
- // force database flush
- ethereum.BlockDb().Close()
- ethereum.StateDb().Close()
- ethereum.ExtraDb().Close()
- fmt.Printf("Import done in %v", time.Since(start))
- return
- }
- func exportchain(ctx *cli.Context) {
- if len(ctx.Args()) != 1 {
- utils.Fatalf("This command requires an argument.")
- }
- cfg := utils.MakeEthConfig(ClientIdentifier, nodeNameVersion, ctx)
- cfg.SkipBcVersionCheck = true
- ethereum, err := eth.New(cfg)
- if err != nil {
- utils.Fatalf("%v\n", err)
- }
- chainmgr := ethereum.ChainManager()
- start := time.Now()
- err = utils.ExportChain(chainmgr, ctx.Args().First())
- if err != nil {
- utils.Fatalf("Export error: %v\n", err)
- }
- fmt.Printf("Export done in %v", time.Since(start))
- return
- }
- func upgradeDb(ctx *cli.Context) {
- fmt.Println("Upgrade blockchain DB")
- cfg := utils.MakeEthConfig(ClientIdentifier, Version, ctx)
- cfg.SkipBcVersionCheck = true
- ethereum, err := eth.New(cfg)
- if err != nil {
- utils.Fatalf("%v\n", err)
- }
- v, _ := ethereum.BlockDb().Get([]byte("BlockchainVersion"))
- bcVersion := int(common.NewValue(v).Uint())
- if bcVersion == 0 {
- bcVersion = core.BlockChainVersion
- }
- filename := fmt.Sprintf("blockchain_%d_%s.chain", bcVersion, time.Now().Format("2006-01-02_15:04:05"))
- exportFile := filepath.Join(ctx.GlobalString(utils.DataDirFlag.Name), filename)
- err = utils.ExportChain(ethereum.ChainManager(), exportFile)
- if err != nil {
- utils.Fatalf("Unable to export chain for reimport %s\n", err)
- }
- ethereum.BlockDb().Close()
- ethereum.StateDb().Close()
- ethereum.ExtraDb().Close()
- os.RemoveAll(filepath.Join(ctx.GlobalString(utils.DataDirFlag.Name), "blockchain"))
- ethereum, err = eth.New(cfg)
- if err != nil {
- utils.Fatalf("%v\n", err)
- }
- ethereum.BlockDb().Put([]byte("BlockchainVersion"), common.NewValue(core.BlockChainVersion).Bytes())
- err = utils.ImportChain(ethereum.ChainManager(), exportFile)
- if err != nil {
- utils.Fatalf("Import error %v (a backup is made in %s, use the import command to import it)\n", err, exportFile)
- }
- // force database flush
- ethereum.BlockDb().Close()
- ethereum.StateDb().Close()
- ethereum.ExtraDb().Close()
- os.Remove(exportFile)
- fmt.Println("Import finished")
- }
- func dump(ctx *cli.Context) {
- chainmgr, _, stateDb := utils.GetChain(ctx)
- for _, arg := range ctx.Args() {
- var block *types.Block
- if hashish(arg) {
- block = chainmgr.GetBlock(common.HexToHash(arg))
- } else {
- num, _ := strconv.Atoi(arg)
- block = chainmgr.GetBlockByNumber(uint64(num))
- }
- if block == nil {
- fmt.Println("{}")
- utils.Fatalf("block not found")
- } else {
- statedb := state.New(block.Root(), stateDb)
- fmt.Printf("%s\n", statedb.Dump())
- }
- }
- }
- func makedag(ctx *cli.Context) {
- args := ctx.Args()
- wrongArgs := func() {
- utils.Fatalf(`Usage: geth makedag <block number> <outputdir>`)
- }
- switch {
- case len(args) == 2:
- blockNum, err := strconv.ParseUint(args[0], 0, 64)
- dir := args[1]
- if err != nil {
- wrongArgs()
- } else {
- dir = filepath.Clean(dir)
- // seems to require a trailing slash
- if !strings.HasSuffix(dir, "/") {
- dir = dir + "/"
- }
- _, err = ioutil.ReadDir(dir)
- if err != nil {
- utils.Fatalf("Can't find dir")
- }
- fmt.Println("making DAG, this could take awhile...")
- ethash.MakeDAG(blockNum, dir)
- }
- default:
- wrongArgs()
- }
- }
- func version(c *cli.Context) {
- fmt.Println(ClientIdentifier)
- fmt.Println("Version:", Version)
- if gitCommit != "" {
- fmt.Println("Git Commit:", gitCommit)
- }
- fmt.Println("Protocol Version:", c.GlobalInt(utils.ProtocolVersionFlag.Name))
- fmt.Println("Network Id:", c.GlobalInt(utils.NetworkIdFlag.Name))
- fmt.Println("Go Version:", runtime.Version())
- fmt.Println("OS:", runtime.GOOS)
- fmt.Printf("GOPATH=%s\n", os.Getenv("GOPATH"))
- fmt.Printf("GOROOT=%s\n", runtime.GOROOT())
- }
- // hashish returns true for strings that look like hashes.
- func hashish(x string) bool {
- _, err := strconv.Atoi(x)
- return err != nil
- }
- func readPassword(prompt string, warnTerm bool) (string, error) {
- if liner.TerminalSupported() {
- lr := liner.NewLiner()
- defer lr.Close()
- return lr.PasswordPrompt(prompt)
- }
- if warnTerm {
- fmt.Println("!! Unsupported terminal, password will be echoed.")
- }
- fmt.Print(prompt)
- input, err := bufio.NewReader(os.Stdin).ReadString('\n')
- fmt.Println()
- return input, err
- }
|