|
|
- package main
-
- import (
- "bufio"
- "encoding/hex"
- "errors"
- "fmt"
- "io"
- "os"
- "strings"
-
- abcicli "github.com/tendermint/abci/client"
- "github.com/tendermint/abci/types"
- "github.com/tendermint/abci/version"
- "github.com/tendermint/tmlibs/log"
- "github.com/urfave/cli"
- )
-
- // Structure for data passed to print response.
- type response struct {
- // generic abci response
- Data []byte
- Code types.CodeType
- Log string
-
- Query *queryResponse
- }
-
- type queryResponse struct {
- Key []byte
- Value []byte
- Height uint64
- Proof []byte
- }
-
- // client is a global variable so it can be reused by the console
- var client abcicli.Client
-
- var logger log.Logger
-
- func main() {
-
- //workaround for the cli library (https://github.com/urfave/cli/issues/565)
- cli.OsExiter = func(_ int) {}
-
- app := cli.NewApp()
- app.Name = "abci-cli"
- app.Usage = "abci-cli [command] [args...]"
- app.Version = version.Version
- app.Flags = []cli.Flag{
- cli.StringFlag{
- Name: "address",
- Value: "tcp://127.0.0.1:46658",
- Usage: "address of application socket",
- },
- cli.StringFlag{
- Name: "abci",
- Value: "socket",
- Usage: "socket or grpc",
- },
- cli.BoolFlag{
- Name: "verbose",
- Usage: "print the command and results as if it were a console session",
- },
- }
- app.Commands = []cli.Command{
- {
- Name: "batch",
- Usage: "Run a batch of abci commands against an application",
- Action: func(c *cli.Context) error {
- return cmdBatch(app, c)
- },
- },
- {
- Name: "console",
- Usage: "Start an interactive abci console for multiple commands",
- Action: func(c *cli.Context) error {
- return cmdConsole(app, c)
- },
- },
- {
- Name: "echo",
- Usage: "Have the application echo a message",
- Action: func(c *cli.Context) error {
- return cmdEcho(c)
- },
- },
- {
- Name: "info",
- Usage: "Get some info about the application",
- Action: func(c *cli.Context) error {
- return cmdInfo(c)
- },
- },
- {
- Name: "set_option",
- Usage: "Set an option on the application",
- Action: func(c *cli.Context) error {
- return cmdSetOption(c)
- },
- },
- {
- Name: "deliver_tx",
- Usage: "Deliver a new tx to application",
- Action: func(c *cli.Context) error {
- return cmdDeliverTx(c)
- },
- },
- {
- Name: "check_tx",
- Usage: "Validate a tx",
- Action: func(c *cli.Context) error {
- return cmdCheckTx(c)
- },
- },
- {
- Name: "commit",
- Usage: "Commit the application state and return the Merkle root hash",
- Action: func(c *cli.Context) error {
- return cmdCommit(c)
- },
- },
- {
- Name: "query",
- Usage: "Query application state",
- Action: func(c *cli.Context) error {
- return cmdQuery(c)
- },
- },
- }
- app.Before = before
- err := app.Run(os.Args)
- if err != nil {
- logger.Error(err.Error())
- os.Exit(1)
- }
-
- }
-
- func before(c *cli.Context) error {
- if logger == nil {
- logger = log.NewTMLogger(log.NewSyncWriter(os.Stdout))
- }
- if client == nil {
- var err error
- client, err = abcicli.NewClient(c.GlobalString("address"), c.GlobalString("abci"), false)
- if err != nil {
- logger.Error(err.Error())
- os.Exit(1)
- }
- client.SetLogger(logger.With("module", "abci-client"))
- }
- return nil
- }
-
- // badCmd is called when we invoke with an invalid first argument (just for console for now)
- func badCmd(c *cli.Context, cmd string) {
- fmt.Println("Unknown command:", cmd)
- fmt.Println("Please try one of the following:")
- fmt.Println("")
- cli.DefaultAppComplete(c)
- }
-
- //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 orginal 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 cmdBatch(app *cli.App, c *cli.Context) error {
- bufReader := bufio.NewReader(os.Stdin)
- for {
- line, more, err := bufReader.ReadLine()
- if more {
- return errors.New("Input line is too long")
- } else if err == io.EOF {
- break
- } else if len(line) == 0 {
- continue
- } else if err != nil {
- return err
- }
-
- args := persistentArgs(line)
- app.Run(args) //cli prints error within its func call
- }
- return nil
- }
-
- func cmdConsole(app *cli.App, c *cli.Context) error {
- // don't hard exit on mistyped commands (eg. check vs check_tx)
- app.CommandNotFound = badCmd
-
- for {
- fmt.Printf("\n> ")
- 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
- }
-
- args := persistentArgs(line)
- app.Run(args) //cli prints error within its func call
- }
- }
-
- // Have the application echo a message
- func cmdEcho(c *cli.Context) error {
- args := c.Args()
- if len(args) != 1 {
- return errors.New("Command echo takes 1 argument")
- }
- resEcho := client.EchoSync(args[0])
- printResponse(c, response{
- Data: resEcho.Data,
- })
- return nil
- }
-
- // Get some info from the application
- func cmdInfo(c *cli.Context) error {
- resInfo, err := client.InfoSync()
- if err != nil {
- return err
- }
- printResponse(c, response{
- Data: []byte(resInfo.Data),
- })
- return nil
- }
-
- // Set an option on the application
- func cmdSetOption(c *cli.Context) error {
- args := c.Args()
- if len(args) != 2 {
- return errors.New("Command set_option takes 2 arguments (key, value)")
- }
- resSetOption := client.SetOptionSync(args[0], args[1])
- printResponse(c, response{
- Log: resSetOption.Log,
- })
- return nil
- }
-
- // Append a new tx to application
- func cmdDeliverTx(c *cli.Context) error {
- args := c.Args()
- if len(args) != 1 {
- return errors.New("Command deliver_tx takes 1 argument")
- }
- txBytes, err := stringOrHexToBytes(c.Args()[0])
- if err != nil {
- return err
- }
- res := client.DeliverTxSync(txBytes)
- printResponse(c, response{
- Code: res.Code,
- Data: res.Data,
- Log: res.Log,
- })
- return nil
- }
-
- // Validate a tx
- func cmdCheckTx(c *cli.Context) error {
- args := c.Args()
- if len(args) != 1 {
- return errors.New("Command check_tx takes 1 argument")
- }
- txBytes, err := stringOrHexToBytes(c.Args()[0])
- if err != nil {
- return err
- }
- res := client.CheckTxSync(txBytes)
- printResponse(c, response{
- Code: res.Code,
- Data: res.Data,
- Log: res.Log,
- })
- return nil
- }
-
- // Get application Merkle root hash
- func cmdCommit(c *cli.Context) error {
- res := client.CommitSync()
- printResponse(c, response{
- Code: res.Code,
- Data: res.Data,
- Log: res.Log,
- })
- return nil
- }
-
- // Query application state
- // TODO: Make request and response support all fields.
- func cmdQuery(c *cli.Context) error {
- args := c.Args()
- if len(args) != 1 {
- return errors.New("Command query takes 1 argument")
- }
- queryBytes, err := stringOrHexToBytes(c.Args()[0])
- if err != nil {
- return err
- }
- resQuery, err := client.QuerySync(types.RequestQuery{
- Data: queryBytes,
- Path: "/store", // TOOD expose
- Height: 0, // TODO expose
- //Prove: true, // TODO expose
- })
- if err != nil {
- return err
- }
- printResponse(c, response{
- Code: resQuery.Code,
- Log: resQuery.Log,
- Query: &queryResponse{
- Key: resQuery.Key,
- Value: resQuery.Value,
- Height: resQuery.Height,
- Proof: resQuery.Proof,
- },
- })
- return nil
- }
-
- //--------------------------------------------------------------------------------
-
- func printResponse(c *cli.Context, rsp response) {
-
- verbose := c.GlobalBool("verbose")
-
- if verbose {
- fmt.Println(">", c.Command.Name, strings.Join(c.Args(), " "))
- }
-
- if !rsp.Code.IsOK() {
- fmt.Printf("-> code: %s\n", rsp.Code.String())
- }
- if len(rsp.Data) != 0 {
- fmt.Printf("-> data: %s\n", rsp.Data)
- fmt.Printf("-> data.hex: %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.Proof != nil {
- fmt.Printf("-> proof: %X\n", rsp.Query.Proof)
- }
- }
-
- if verbose {
- fmt.Println("")
- }
-
- }
-
- // 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
- }
|