package main import ( "bufio" "encoding/hex" "errors" "fmt" "io" "os" "os/signal" "strings" "syscall" "github.com/spf13/cobra" "github.com/tendermint/tendermint/libs/log" "github.com/tendermint/tendermint/version" abciclient "github.com/tendermint/tendermint/abci/client" "github.com/tendermint/tendermint/abci/example/code" "github.com/tendermint/tendermint/abci/example/kvstore" "github.com/tendermint/tendermint/abci/server" servertest "github.com/tendermint/tendermint/abci/tests/server" "github.com/tendermint/tendermint/abci/types" "github.com/tendermint/tendermint/proto/tendermint/crypto" ) // client is a global variable so it can be reused by the console var ( client abciclient.Client ) // flags var ( // global flagAddress string flagAbci string flagVerbose bool // for the println output flagLogLevel string // for the logger // query flagPath string flagHeight int flagProve bool // kvstore flagPersist string ) func RootCmmand(logger log.Logger) *cobra.Command { return &cobra.Command{ Use: "abci-cli", Short: "the ABCI CLI tool wraps an ABCI client", Long: "the ABCI CLI tool wraps an ABCI client and is used for testing ABCI servers", PersistentPreRunE: func(cmd *cobra.Command, args []string) (err error) { switch cmd.Use { case "kvstore", "version": return nil } if client == nil { var err error client, err = abciclient.NewClient(logger.With("module", "abci-client"), flagAddress, flagAbci, false) if err != nil { return err } if err := client.Start(cmd.Context()); err != nil { return err } } return nil }, } } // Structure for data passed to print response. type response struct { // generic abci response Data []byte Code uint32 Info string Log string Query *queryResponse } type queryResponse struct { Key []byte Value []byte Height int64 ProofOps *crypto.ProofOps } func Execute() error { logger, err := log.NewDefaultLogger(log.LogFormatPlain, log.LogLevelInfo) if err != nil { return err } cmd := RootCmmand(logger) addGlobalFlags(cmd) addCommands(cmd, logger) return cmd.Execute() } func addGlobalFlags(cmd *cobra.Command) { cmd.PersistentFlags().StringVarP(&flagAddress, "address", "", "tcp://0.0.0.0:26658", "address of application socket") cmd.PersistentFlags().StringVarP(&flagAbci, "abci", "", "socket", "either socket or grpc") cmd.PersistentFlags().BoolVarP(&flagVerbose, "verbose", "v", false, "print the command and results as if it were a console session") cmd.PersistentFlags().StringVarP(&flagLogLevel, "log_level", "", "debug", "set the logger level") } func addCommands(cmd *cobra.Command, logger log.Logger) { cmd.AddCommand(batchCmd) cmd.AddCommand(consoleCmd) cmd.AddCommand(echoCmd) cmd.AddCommand(infoCmd) cmd.AddCommand(finalizeBlockCmd) cmd.AddCommand(checkTxCmd) cmd.AddCommand(commitCmd) cmd.AddCommand(versionCmd) cmd.AddCommand(testCmd) cmd.AddCommand(getQueryCmd()) // examples cmd.AddCommand(getKVStoreCmd(logger)) } var batchCmd = &cobra.Command{ Use: "batch", Short: "run a batch of abci commands against an application", Long: `run a batch of abci commands against an application This command is run by piping in a file containing a series of commands you'd like to run: abci-cli batch < example.file where example.file looks something like: check_tx 0x00 check_tx 0xff finalize_block 0x00 check_tx 0x00 finalize_block 0x01 0x04 0xff info `, Args: cobra.ExactArgs(0), RunE: cmdBatch, } var consoleCmd = &cobra.Command{ Use: "console", Short: "start an interactive ABCI console for multiple commands", Long: `start an interactive ABCI console for multiple commands This command opens an interactive console for running any of the other commands without opening a new connection each time `, Args: cobra.ExactArgs(0), ValidArgs: []string{"echo", "info", "finalize_block", "check_tx", "commit", "query"}, RunE: cmdConsole, } var echoCmd = &cobra.Command{ Use: "echo", Short: "have the application echo a message", Long: "have the application echo a message", Args: cobra.ExactArgs(1), RunE: cmdEcho, } var infoCmd = &cobra.Command{ Use: "info", Short: "get some info about the application", Long: "get some info about the application", Args: cobra.ExactArgs(0), RunE: cmdInfo, } var finalizeBlockCmd = &cobra.Command{ Use: "finalize_block", Short: "deliver a block of transactions to the application", Long: "deliver a block of transactions to the application", Args: cobra.MinimumNArgs(1), RunE: cmdFinalizeBlock, } var checkTxCmd = &cobra.Command{ Use: "check_tx", Short: "validate a transaction", Long: "validate a transaction", Args: cobra.ExactArgs(1), RunE: cmdCheckTx, } var commitCmd = &cobra.Command{ Use: "commit", Short: "commit the application state and return the Merkle root hash", Long: "commit the application state and return the Merkle root hash", Args: cobra.ExactArgs(0), RunE: cmdCommit, } var versionCmd = &cobra.Command{ Use: "version", Short: "print ABCI console version", Long: "print ABCI console version", Args: cobra.ExactArgs(0), RunE: func(cmd *cobra.Command, args []string) error { fmt.Println(version.ABCIVersion) return nil }, } func getQueryCmd() *cobra.Command { cmd := &cobra.Command{ Use: "query", Short: "query the application state", Long: "query the application state", Args: cobra.ExactArgs(1), RunE: cmdQuery, } cmd.PersistentFlags().StringVarP(&flagPath, "path", "", "/store", "path to prefix query with") cmd.PersistentFlags().IntVarP(&flagHeight, "height", "", 0, "height to query the blockchain at") cmd.PersistentFlags().BoolVarP(&flagProve, "prove", "", false, "whether or not to return a merkle proof of the query result") return cmd } func getKVStoreCmd(logger log.Logger) *cobra.Command { cmd := &cobra.Command{ Use: "kvstore", Short: "ABCI demo example", Long: "ABCI demo example", Args: cobra.ExactArgs(0), RunE: makeKVStoreCmd(logger), } cmd.PersistentFlags().StringVarP(&flagPersist, "persist", "", "", "directory to use for a database") return cmd } var testCmd = &cobra.Command{ Use: "test", Short: "run integration tests", Long: "run integration tests", Args: cobra.ExactArgs(0), RunE: cmdTest, } // Generates new Args array based off of previous call args to maintain flag persistence func persistentArgs(line []byte) []string { // generate the arguments to run from original os.Args // to maintain flag arguments args := os.Args args = args[:len(args)-1] // remove the previous command argument if len(line) > 0 { // prevents introduction of extra space leading to argument parse errors args = append(args, strings.Split(string(line), " ")...) } return args } //-------------------------------------------------------------------------------- func compose(fs []func() error) error { if len(fs) == 0 { return nil } err := fs[0]() if err == nil { return compose(fs[1:]) } return err } func cmdTest(cmd *cobra.Command, args []string) error { ctx := cmd.Context() return compose( []func() error{ func() error { return servertest.InitChain(ctx, client) }, func() error { return servertest.Commit(ctx, client, nil) }, func() error { return servertest.FinalizeBlock(ctx, client, [][]byte{ []byte("abc"), }, []uint32{ code.CodeTypeBadNonce, }, nil) }, func() error { return servertest.Commit(ctx, client, nil) }, func() error { return servertest.FinalizeBlock(ctx, client, [][]byte{ {0x00}, }, []uint32{ code.CodeTypeOK, }, nil) }, func() error { return servertest.Commit(ctx, client, []byte{0, 0, 0, 0, 0, 0, 0, 1}) }, func() error { return servertest.FinalizeBlock(ctx, client, [][]byte{ {0x00}, {0x01}, {0x00, 0x02}, {0x00, 0x03}, {0x00, 0x00, 0x04}, {0x00, 0x00, 0x06}, }, []uint32{ code.CodeTypeBadNonce, code.CodeTypeOK, code.CodeTypeOK, code.CodeTypeOK, code.CodeTypeOK, code.CodeTypeBadNonce, }, nil) }, func() error { return servertest.Commit(ctx, client, []byte{0, 0, 0, 0, 0, 0, 0, 5}) }, }) } func cmdBatch(cmd *cobra.Command, args []string) error { bufReader := bufio.NewReader(os.Stdin) LOOP: for { line, more, err := bufReader.ReadLine() switch { case more: return errors.New("input line is too long") case err == io.EOF: break LOOP case len(line) == 0: continue case err != nil: return err } cmdArgs := persistentArgs(line) if err := muxOnCommands(cmd, cmdArgs); err != nil { return err } fmt.Println() } return nil } func cmdConsole(cmd *cobra.Command, args []string) error { for { fmt.Printf("> ") bufReader := bufio.NewReader(os.Stdin) line, more, err := bufReader.ReadLine() if more { return errors.New("input is too long") } else if err != nil { return err } pArgs := persistentArgs(line) if err := muxOnCommands(cmd, pArgs); err != nil { return err } } } func muxOnCommands(cmd *cobra.Command, pArgs []string) error { if len(pArgs) < 2 { return errors.New("expecting persistent args of the form: abci-cli [command] <...>") } // TODO: this parsing is fragile args := []string{} for i := 0; i < len(pArgs); i++ { arg := pArgs[i] // check for flags if strings.HasPrefix(arg, "-") { // if it has an equal, we can just skip if strings.Contains(arg, "=") { continue } // if its a boolean, we can just skip _, err := cmd.Flags().GetBool(strings.TrimLeft(arg, "-")) if err == nil { continue } // otherwise, we need to skip the next one too i++ continue } // append the actual arg args = append(args, arg) } var subCommand string var actualArgs []string if len(args) > 1 { subCommand = args[1] } if len(args) > 2 { actualArgs = args[2:] } cmd.Use = subCommand // for later print statements ... switch strings.ToLower(subCommand) { case "check_tx": return cmdCheckTx(cmd, actualArgs) case "commit": return cmdCommit(cmd, actualArgs) case "finalize_block": return cmdFinalizeBlock(cmd, actualArgs) case "echo": return cmdEcho(cmd, actualArgs) case "info": return cmdInfo(cmd, actualArgs) case "query": return cmdQuery(cmd, actualArgs) default: return cmdUnimplemented(cmd, pArgs) } } func cmdUnimplemented(cmd *cobra.Command, args []string) error { msg := "unimplemented command" if len(args) > 0 { msg += fmt.Sprintf(" args: [%s]", strings.Join(args, " ")) } printResponse(cmd, args, response{ Code: codeBad, Log: msg, }) fmt.Println("Available commands:") for _, cmd := range cmd.Commands() { fmt.Printf("%s: %s\n", cmd.Use, cmd.Short) } fmt.Println("Use \"[command] --help\" for more information about a command.") return nil } // Have the application echo a message func cmdEcho(cmd *cobra.Command, args []string) error { msg := "" if len(args) > 0 { msg = args[0] } res, err := client.Echo(cmd.Context(), msg) if err != nil { return err } printResponse(cmd, args, response{ Data: []byte(res.Message), }) return nil } // Get some info from the application func cmdInfo(cmd *cobra.Command, args []string) error { var version string if len(args) == 1 { version = args[0] } res, err := client.Info(cmd.Context(), types.RequestInfo{Version: version}) if err != nil { return err } printResponse(cmd, args, response{ Data: []byte(res.Data), }) return nil } const codeBad uint32 = 10 // Append a new tx to application func cmdFinalizeBlock(cmd *cobra.Command, args []string) error { if len(args) == 0 { printResponse(cmd, args, response{ Code: codeBad, Log: "Must provide at least one transaction", }) return nil } txs := make([][]byte, len(args)) for i, arg := range args { txBytes, err := stringOrHexToBytes(arg) if err != nil { return err } txs[i] = txBytes } res, err := client.FinalizeBlock(cmd.Context(), types.RequestFinalizeBlock{Txs: txs}) if err != nil { return err } for _, tx := range res.TxResults { printResponse(cmd, args, response{ Code: tx.Code, Data: tx.Data, Info: tx.Info, Log: tx.Log, }) } return nil } // Validate a tx func cmdCheckTx(cmd *cobra.Command, args []string) error { if len(args) == 0 { printResponse(cmd, args, response{ Code: codeBad, Info: "want the tx", }) return nil } txBytes, err := stringOrHexToBytes(args[0]) if err != nil { return err } res, err := client.CheckTx(cmd.Context(), types.RequestCheckTx{Tx: txBytes}) if err != nil { return err } printResponse(cmd, args, response{ Code: res.Code, Data: res.Data, Info: res.Info, Log: res.Log, }) return nil } // Get application Merkle root hash func cmdCommit(cmd *cobra.Command, args []string) error { res, err := client.Commit(cmd.Context()) if err != nil { return err } printResponse(cmd, args, response{ Data: res.Data, }) return nil } // Query application state func cmdQuery(cmd *cobra.Command, args []string) error { if len(args) == 0 { printResponse(cmd, args, response{ Code: codeBad, Info: "want the query", Log: "", }) return nil } queryBytes, err := stringOrHexToBytes(args[0]) if err != nil { return err } resQuery, err := client.Query(cmd.Context(), types.RequestQuery{ Data: queryBytes, Path: flagPath, Height: int64(flagHeight), Prove: flagProve, }) if err != nil { return err } printResponse(cmd, args, response{ Code: resQuery.Code, Info: resQuery.Info, Log: resQuery.Log, Query: &queryResponse{ Key: resQuery.Key, Value: resQuery.Value, Height: resQuery.Height, ProofOps: resQuery.ProofOps, }, }) return nil } func makeKVStoreCmd(logger log.Logger) func(*cobra.Command, []string) error { return func(cmd *cobra.Command, args []string) error { // Create the application - in memory or persisted to disk var app types.Application if flagPersist == "" { app = kvstore.NewApplication() } else { app = kvstore.NewPersistentKVStoreApplication(logger, flagPersist) } // Start the listener srv, err := server.NewServer(logger.With("module", "abci-server"), flagAddress, flagAbci, app) if err != nil { return err } ctx, cancel := signal.NotifyContext(cmd.Context(), syscall.SIGTERM) defer cancel() if err := srv.Start(ctx); err != nil { return err } // Run forever. <-ctx.Done() return nil } } //-------------------------------------------------------------------------------- func printResponse(cmd *cobra.Command, args []string, rsp response) { if flagVerbose { fmt.Println(">", cmd.Use, strings.Join(args, " ")) } // Always print the status code. if rsp.Code == types.CodeTypeOK { fmt.Printf("-> code: OK\n") } else { fmt.Printf("-> code: %d\n", rsp.Code) } if len(rsp.Data) != 0 { // Do no print this line when using the commit command // because the string comes out as gibberish if cmd.Use != "commit" { fmt.Printf("-> data: %s\n", rsp.Data) } fmt.Printf("-> data.hex: 0x%X\n", rsp.Data) } if rsp.Log != "" { fmt.Printf("-> log: %s\n", rsp.Log) } if rsp.Query != nil { fmt.Printf("-> height: %d\n", rsp.Query.Height) if rsp.Query.Key != nil { fmt.Printf("-> key: %s\n", rsp.Query.Key) fmt.Printf("-> key.hex: %X\n", rsp.Query.Key) } if rsp.Query.Value != nil { fmt.Printf("-> value: %s\n", rsp.Query.Value) fmt.Printf("-> value.hex: %X\n", rsp.Query.Value) } if rsp.Query.ProofOps != nil { fmt.Printf("-> proof: %#v\n", rsp.Query.ProofOps) } } } // NOTE: s is interpreted as a string unless prefixed with 0x func stringOrHexToBytes(s string) ([]byte, error) { if len(s) > 2 && strings.ToLower(s[:2]) == "0x" { b, err := hex.DecodeString(s[2:]) if err != nil { err = fmt.Errorf("error decoding hex argument: %s", err.Error()) return nil, err } return b, nil } if !strings.HasPrefix(s, "\"") || !strings.HasSuffix(s, "\"") { err := fmt.Errorf("invalid string arg: \"%s\". Must be quoted or a \"0x\"-prefixed hex string", s) return nil, err } return []byte(s[1 : len(s)-1]), nil }