| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333 |
- /*
- 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"
- "os"
- "runtime"
- "strconv"
- "strings"
- "time"
- "github.com/codegangsta/cli"
- "github.com/ethereum/go-ethereum/cmd/utils"
- "github.com/ethereum/go-ethereum/core/types"
- "github.com/ethereum/go-ethereum/eth"
- "github.com/ethereum/go-ethereum/ethutil"
- "github.com/ethereum/go-ethereum/logger"
- "github.com/ethereum/go-ethereum/state"
- "github.com/peterh/liner"
- )
- const (
- ClientIdentifier = "Ethereum(G)"
- Version = "0.9.0"
- )
- var (
- clilogger = logger.NewLogger("CLI")
- app = utils.NewApp(Version, "the go-ethereum command line interface")
- )
- func init() {
- app.Action = run
- app.HideVersion = true // we have a command to print the version
- app.Commands = []cli.Command{
- {
- Action: version,
- Name: "version",
- Usage: "print ethereum version numbers",
- Description: `
- The output of this command is supposed to be machine-readable.
- `,
- },
- {
- Action: accountList,
- Name: "account",
- Usage: "manage accounts",
- Subcommands: []cli.Command{
- {
- Action: accountList,
- Name: "list",
- Usage: "print account addresses",
- },
- {
- Action: accountCreate,
- Name: "new",
- Usage: "create a new account",
- },
- },
- },
- {
- 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: runjs,
- Name: "js",
- Usage: `interactive JavaScript console`,
- Description: `
- In the console, you can use the eth object to interact
- with the running ethereum stack. The API does not match
- ethereum.js.
- A JavaScript file can be provided as the argument. The
- runtime will execute the file and exit.
- `,
- },
- {
- Action: importchain,
- Name: "import",
- Usage: `import a blockchain file`,
- },
- {
- Action: exportchain,
- Name: "export",
- Usage: `export blockchain into file`,
- },
- }
- app.Flags = []cli.Flag{
- utils.UnlockedAccountFlag,
- utils.BootnodesFlag,
- utils.DataDirFlag,
- utils.ListenPortFlag,
- utils.LogFileFlag,
- utils.LogFormatFlag,
- utils.LogLevelFlag,
- utils.MaxPeersFlag,
- utils.MinerThreadsFlag,
- utils.MiningEnabledFlag,
- utils.NATFlag,
- utils.NodeKeyFileFlag,
- utils.NodeKeyHexFlag,
- utils.RPCEnabledFlag,
- utils.RPCListenAddrFlag,
- utils.RPCPortFlag,
- utils.UnencryptedKeysFlag,
- utils.VMDebugFlag,
- //utils.VMTypeFlag,
- }
- // 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")
- // potential subcommands:
- // flag.StringVar(&SecretFile, "import", "", "imports the file given (hex or mnemonic formats)")
- // flag.StringVar(&ExportDir, "export", "", "exports the session keyring to files in the directory given")
- // flag.BoolVar(&GenAddr, "genaddr", false, "create a new priv/pub key")
- }
- func main() {
- 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) {
- fmt.Printf("Welcome to the FRONTIER\n")
- utils.HandleInterrupt()
- cfg := utils.MakeEthConfig(ClientIdentifier, Version, ctx)
- ethereum, err := eth.New(cfg)
- if err != nil {
- utils.Fatalf("%v", err)
- }
- startEth(ctx, ethereum)
- // this blocks the thread
- ethereum.WaitForShutdown()
- }
- func runjs(ctx *cli.Context) {
- cfg := utils.MakeEthConfig(ClientIdentifier, Version, ctx)
- ethereum, err := eth.New(cfg)
- if err != nil {
- utils.Fatalf("%v", err)
- }
- startEth(ctx, ethereum)
- repl := newJSRE(ethereum)
- if len(ctx.Args()) == 0 {
- repl.interactive()
- } else {
- for _, file := range ctx.Args() {
- repl.exec(file)
- }
- }
- ethereum.Stop()
- ethereum.WaitForShutdown()
- }
- func startEth(ctx *cli.Context, eth *eth.Ethereum) {
- utils.StartEthereum(eth)
- // Load startup keys. XXX we are going to need a different format
- account := ctx.GlobalString(utils.UnlockedAccountFlag.Name)
- if len(account) > 0 {
- split := strings.Split(account, ":")
- if len(split) != 2 {
- utils.Fatalf("Illegal 'unlock' format (address:password)")
- }
- am := eth.AccountManager()
- // Attempt to unlock the account
- err := am.Unlock(ethutil.Hex2Bytes(split[0]), split[1])
- if err != nil {
- utils.Fatalf("Unlock account failed '%v'", err)
- }
- }
- // Start auxiliary services if enabled.
- if ctx.GlobalBool(utils.RPCEnabledFlag.Name) {
- utils.StartRPC(eth, ctx)
- }
- if ctx.GlobalBool(utils.MiningEnabledFlag.Name) {
- eth.StartMining()
- }
- }
- func accountList(ctx *cli.Context) {
- am := utils.GetAccountManager(ctx)
- accts, err := am.Accounts()
- if err != nil {
- utils.Fatalf("Could not list accounts: %v", err)
- }
- for _, acct := range accts {
- fmt.Printf("Address: %#x\n", acct)
- }
- }
- func accountCreate(ctx *cli.Context) {
- am := utils.GetAccountManager(ctx)
- passphrase := ""
- if !ctx.GlobalBool(utils.UnencryptedKeysFlag.Name) {
- fmt.Println("The new account will be encrypted with a passphrase.")
- fmt.Println("Please enter a passphrase now.")
- auth, err := readPassword("Passphrase: ", true)
- if err != nil {
- utils.Fatalf("%v", err)
- }
- confirm, err := readPassword("Repeat Passphrase: ", false)
- if err != nil {
- utils.Fatalf("%v", err)
- }
- if auth != confirm {
- utils.Fatalf("Passphrases did not match.")
- }
- passphrase = auth
- }
- acct, err := am.NewAccount(passphrase)
- if err != nil {
- utils.Fatalf("Could not create the account: %v", err)
- }
- fmt.Printf("Address: %#x\n", acct.Address)
- }
- func importchain(ctx *cli.Context) {
- if len(ctx.Args()) != 1 {
- utils.Fatalf("This command requires an argument.")
- }
- chainmgr, _, _ := utils.GetChain(ctx)
- start := time.Now()
- err := utils.ImportChain(chainmgr, ctx.Args().First())
- if err != nil {
- utils.Fatalf("Import error: %v\n", err)
- }
- 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.")
- }
- chainmgr, _, _ := utils.GetChain(ctx)
- 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 dump(ctx *cli.Context) {
- chainmgr, _, stateDb := utils.GetChain(ctx)
- for _, arg := range ctx.Args() {
- var block *types.Block
- if hashish(arg) {
- block = chainmgr.GetBlock(ethutil.Hex2Bytes(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())
- // fmt.Println(block)
- }
- }
- }
- func version(c *cli.Context) {
- fmt.Printf(`%v
- Version: %v
- Protocol Version: %d
- Network Id: %d
- GO: %s
- OS: %s
- GOPATH=%s
- GOROOT=%s
- `, ClientIdentifier, Version, eth.ProtocolVersion, eth.NetworkId, runtime.Version(), runtime.GOOS, os.Getenv("GOPATH"), 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
- }
|