package main import ( "context" "encoding/json" "errors" "fmt" "io" "io/ioutil" "os" "os/signal" "runtime" "sort" "syscall" "github.com/ledgerwatch/turbo-geth/cmd/utils" "github.com/ledgerwatch/turbo-geth/console/prompt" "github.com/ledgerwatch/turbo-geth/crypto" "github.com/ledgerwatch/turbo-geth/eth" "github.com/ledgerwatch/turbo-geth/internal/debug" "github.com/ledgerwatch/turbo-geth/log" "github.com/ledgerwatch/turbo-geth/p2p" "github.com/ledgerwatch/turbo-geth/p2p/enode" "github.com/mattn/go-colorable" "github.com/mattn/go-isatty" "github.com/urfave/cli" ) var ( // Git SHA1 commit hash of the release (set via linker flags) gitCommit = "" gitDate = "" // The app that holds all commands and flags. app = utils.NewApp(gitCommit, gitDate, "Ethereum Tester") // flags that configure the node VerbosityFlag = cli.IntFlag{ Name: "verbosity", Usage: "Logging verbosity: 0=silent, 1=error, 2=warn, 3=info, 4=debug, 5=detail", Value: 3, } flags = []cli.Flag{VerbosityFlag} ) func init() { // Initialize the CLI app and start TurboGeth app.Action = tester app.HideVersion = true // we have a command to print the version app.Copyright = "Copyright 2020 The turbo-geth Authors" app.Commands = []cli.Command{} sort.Sort(cli.CommandsByName(app.Commands)) app.Flags = append(app.Flags, flags...) app.Before = func(ctx *cli.Context) error { setupLogger(ctx) runtime.GOMAXPROCS(runtime.NumCPU()) if err := debug.Setup(ctx); err != nil { return err } return nil } app.After = func(ctx *cli.Context) error { debug.Exit() prompt.Stdin.Close() // Resets terminal mode. return nil } app.Commands = []cli.Command{ { Action: utils.MigrateFlags(genesisCmd), Name: "genesis", Usage: "Produce genesis.json file for geth", }, } } func main() { if err := app.Run(os.Args); err != nil { fmt.Fprintln(os.Stderr, err) os.Exit(1) } } func rootContext() context.Context { ctx, cancel := context.WithCancel(context.Background()) go func() { ch := make(chan os.Signal, 1) signal.Notify(ch, os.Interrupt, syscall.SIGTERM) defer signal.Stop(ch) select { case <-ch: log.Info("Got interrupt, shutting down...") case <-ctx.Done(): } cancel() }() return ctx } func setupLogger(cliCtx *cli.Context) { var ( ostream log.Handler glogger *log.GlogHandler ) usecolor := (isatty.IsTerminal(os.Stderr.Fd()) || isatty.IsCygwinTerminal(os.Stderr.Fd())) && os.Getenv("TERM") != "dumb" output := io.Writer(os.Stderr) if usecolor { output = colorable.NewColorableStderr() } ostream = log.StreamHandler(output, log.TerminalFormat(usecolor)) glogger = log.NewGlogHandler(ostream) log.Root().SetHandler(glogger) glogger.Verbosity(log.Lvl(cliCtx.GlobalInt(VerbosityFlag.Name))) } func tester(cliCtx *cli.Context) error { ctx := rootContext() nodeToConnect, err := getTargetAddr(cliCtx) if err != nil { return err } tp := NewTesterProtocol() //fmt.Printf("%s %s\n", ctx.Args()[0], ctx.Args()[1]) //tp.blockFeeder, err = NewBlockAccessor(ctx.Args()[0]/*, ctx.Args()[1]*/) blockGen, err := NewBlockGenerator(ctx, "blocks", 50000) if err != nil { panic(fmt.Sprintf("Failed to create block generator: %v", err)) } defer blockGen.Close() tp.blockFeeder = blockGen tp.forkBase = 49998 tp.forkHeight = 5 tp.forkFeeder, err = NewForkGenerator(ctx, blockGen, "forkblocks", tp.forkBase, tp.forkHeight) if err != nil { panic(fmt.Sprintf("Failed to create fork generator: %v", err)) } defer tp.forkFeeder.Close() tp.protocolVersion = uint32(eth.ProtocolVersions[0]) tp.networkId = 1 // Mainnet tp.genesisBlockHash = tp.forkFeeder.Genesis().Hash() server := makeP2PServer(ctx, tp, []string{eth.ProtocolName, eth.DebugName}) // Add protocol if err := server.Start(); err != nil { panic(fmt.Errorf("could not start server: %w", err)) } server.AddPeer(nodeToConnect) <-ctx.Done() return nil } func genesisCmd(cliCtx *cli.Context) error { res, err := json.Marshal(genesis()) if err != nil { return err } _, err = fmt.Fprintf(os.Stdout, string(res)) if err != nil { return err } return nil } func makeP2PServer(ctx context.Context, tp *TesterProtocol, protocols []string) *p2p.Server { serverKey, err := crypto.GenerateKey() if err != nil { panic(fmt.Sprintf("Failed to generate server key: %v", err)) } p2pConfig := p2p.Config{} p2pConfig.PrivateKey = serverKey p2pConfig.Name = "geth tester" p2pConfig.Logger = log.New() p2pConfig.MaxPeers = 1 p2pConfig.Protocols = []p2p.Protocol{} pMap := map[string]p2p.Protocol{ eth.ProtocolName: { Name: eth.ProtocolName, Version: eth.ProtocolVersions[0], Length: eth.ProtocolLengths[eth.ProtocolVersions[0]], Run: func(peer *p2p.Peer, rw p2p.MsgReadWriter) error { return tp.protocolRun(ctx, peer, rw) }, }, eth.DebugName: { Name: eth.DebugName, Version: eth.DebugVersions[0], Length: eth.DebugLengths[eth.DebugVersions[0]], Run: func(peer *p2p.Peer, rw p2p.MsgReadWriter) error { return tp.debugProtocolRun(ctx, peer, rw) }, }, } for _, protocolName := range protocols { p2pConfig.Protocols = append(p2pConfig.Protocols, pMap[protocolName]) } return &p2p.Server{Config: p2pConfig} } func getTargetAddr(cliCtx *cli.Context) (*enode.Node, error) { var enodeAddress string if len(cliCtx.Args()) < 1 { addr, err := ioutil.ReadFile(p2p.EnodeAddressFileName) if err != nil { return nil, err } enodeAddress = string(addr) } else { enodeAddress = cliCtx.Args()[0] } if enodeAddress == "" { return nil, errors.New("Usage: tester \n") } nodeToConnect, err := enode.ParseV4(enodeAddress) if err != nil { return nil, fmt.Errorf("could not parse the node info: %w", err) } fmt.Printf("Parsed node: %s, IP: %s\n", nodeToConnect, nodeToConnect.IP()) return nodeToConnect, nil }