tendermint/cmd/tmsp-cli/tmsp-cli.go

357 lines
7.9 KiB
Go
Raw Normal View History

2015-11-02 07:39:53 -08:00
package main
import (
2015-11-30 17:56:36 -08:00
"bufio"
2016-07-01 17:22:58 -07:00
"encoding/hex"
"errors"
2015-11-02 07:39:53 -08:00
"fmt"
2015-11-30 22:49:54 -08:00
"io"
2015-11-02 07:39:53 -08:00
"os"
2015-11-30 17:56:36 -08:00
"strings"
2015-11-02 07:39:53 -08:00
. "github.com/tendermint/go-common"
2016-05-14 00:09:47 -07:00
"github.com/tendermint/tmsp/client"
2015-11-02 07:39:53 -08:00
"github.com/tendermint/tmsp/types"
2016-10-20 18:57:33 -07:00
"github.com/urfave/cli"
2015-11-02 07:39:53 -08:00
)
2017-01-11 22:03:51 -08:00
//structure for data passed to print response
// variables must be exposed for JSON to read
type response struct {
Res types.Result
Data string
PrintCode bool
Code string
}
func newResponse(res types.Result, data string, printCode bool) *response {
rsp := &response{
Res: res,
Data: data,
PrintCode: printCode,
Code: "",
}
if printCode {
rsp.Code = res.Code.String()
}
return rsp
}
2016-05-24 18:51:28 -07:00
// client is a global variable so it can be reused by the console
2016-05-14 00:09:47 -07:00
var client tmspcli.Client
2015-11-30 17:56:36 -08:00
2015-11-02 07:39:53 -08:00
func main() {
2016-12-22 16:00:22 -08:00
//workaround for the cli library (https://github.com/urfave/cli/issues/565)
cli.OsExiter = func(_ int) {}
2015-11-02 07:39:53 -08:00
app := cli.NewApp()
2016-01-22 15:50:11 -08:00
app.Name = "tmsp-cli"
app.Usage = "tmsp-cli [command] [args...]"
2016-09-12 09:23:36 -07:00
app.Version = "0.2.1" // better error handling in console
2015-11-02 07:39:53 -08:00
app.Flags = []cli.Flag{
cli.StringFlag{
Name: "address",
2015-12-01 15:27:50 -08:00
Value: "tcp://127.0.0.1:46658",
2015-11-02 07:39:53 -08:00
Usage: "address of application socket",
},
2016-05-18 15:30:38 -07:00
cli.StringFlag{
Name: "tmsp",
Value: "socket",
Usage: "socket or grpc",
},
cli.BoolFlag{
Name: "verbose",
Usage: "print the command and results as if it were a console session",
},
2016-05-18 15:30:38 -07:00
}
2015-11-02 07:39:53 -08:00
app.Commands = []cli.Command{
2015-11-30 22:49:54 -08:00
{
Name: "batch",
Usage: "Run a batch of tmsp commands against an application",
Action: func(c *cli.Context) error {
return cmdBatch(app, c)
2015-11-30 22:49:54 -08:00
},
},
2015-11-30 17:56:36 -08:00
{
Name: "console",
Usage: "Start an interactive tmsp console for multiple commands",
Action: func(c *cli.Context) error {
return cmdConsole(app, c)
2015-11-30 17:56:36 -08:00
},
},
2015-12-06 15:18:13 -08:00
{
Name: "echo",
Usage: "Have the application echo a message",
Action: func(c *cli.Context) error {
return cmdEcho(c)
2015-12-06 15:18:13 -08:00
},
},
{
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)
},
},
2015-11-02 07:39:53 -08:00
{
2017-01-12 12:27:08 -08:00
Name: "deliver_tx",
Usage: "Deliver a new tx to application",
Action: func(c *cli.Context) error {
2017-01-12 12:27:08 -08:00
return cmdDeliverTx(c)
2015-11-02 07:39:53 -08:00
},
},
{
Name: "check_tx",
Usage: "Validate a tx",
Action: func(c *cli.Context) error {
return cmdCheckTx(c)
2015-11-02 07:39:53 -08:00
},
},
{
Name: "commit",
Usage: "Commit the application state and return the Merkle root hash",
Action: func(c *cli.Context) error {
return cmdCommit(c)
2015-11-02 07:39:53 -08:00
},
},
2016-01-22 15:50:11 -08:00
{
Name: "query",
Usage: "Query application state",
Action: func(c *cli.Context) error {
return cmdQuery(c)
2016-01-22 15:50:11 -08:00
},
},
2015-11-02 07:39:53 -08:00
}
2015-11-30 17:56:36 -08:00
app.Before = before
err := app.Run(os.Args)
if err != nil {
Exit(err.Error())
}
2015-11-02 07:39:53 -08:00
}
2015-11-30 17:56:36 -08:00
func before(c *cli.Context) error {
2016-05-14 00:09:47 -07:00
if client == nil {
2015-11-30 17:56:36 -08:00
var err error
2016-05-18 15:30:38 -07:00
client, err = tmspcli.NewClient(c.GlobalString("address"), c.GlobalString("tmsp"), false)
2015-11-30 17:56:36 -08:00
if err != nil {
Exit(err.Error())
}
}
return nil
}
2016-09-12 01:21:20 -07:00
// 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)
}
2017-01-11 22:03:51 -08:00
//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
}
2015-11-02 07:39:53 -08:00
//--------------------------------------------------------------------------------
func cmdBatch(app *cli.App, c *cli.Context) error {
2015-11-30 22:49:54 -08:00
bufReader := bufio.NewReader(os.Stdin)
for {
line, more, err := bufReader.ReadLine()
if more {
return errors.New("Input line is too long")
2015-11-30 22:49:54 -08:00
} else if err == io.EOF {
break
2015-12-14 15:00:18 -08:00
} else if len(line) == 0 {
continue
2015-11-30 22:49:54 -08:00
} else if err != nil {
return err
2015-11-30 22:49:54 -08:00
}
2017-01-11 22:03:51 -08:00
args := persistentArgs(line)
app.Run(args) //cli prints error within its func call
2015-11-30 22:49:54 -08:00
}
return nil
2015-11-30 22:49:54 -08:00
}
func cmdConsole(app *cli.App, c *cli.Context) error {
2016-09-12 01:21:20 -07:00
// don't hard exit on mistyped commands (eg. check vs check_tx)
app.CommandNotFound = badCmd
2017-01-11 22:03:51 -08:00
2015-11-30 17:56:36 -08:00
for {
fmt.Printf("\n> ")
2015-11-30 17:56:36 -08:00
bufReader := bufio.NewReader(os.Stdin)
line, more, err := bufReader.ReadLine()
if more {
return errors.New("Input is too long")
2015-11-30 17:56:36 -08:00
} else if err != nil {
return err
2015-11-30 17:56:36 -08:00
}
2017-01-11 22:03:51 -08:00
args := persistentArgs(line)
app.Run(args) //cli prints error within its func call
2015-11-30 17:56:36 -08:00
}
}
2015-12-06 15:18:13 -08:00
// Have the application echo a message
func cmdEcho(c *cli.Context) error {
2015-12-06 15:18:13 -08:00
args := c.Args()
if len(args) != 1 {
return errors.New("Command echo takes 1 argument")
2015-12-06 15:18:13 -08:00
}
2016-05-14 00:09:47 -07:00
res := client.EchoSync(args[0])
2017-01-11 22:03:51 -08:00
rsp := newResponse(res, string(res.Data), false)
printResponse(c, rsp)
return nil
2015-12-06 15:18:13 -08:00
}
// Get some info from the application
func cmdInfo(c *cli.Context) error {
2016-12-26 17:44:36 -08:00
resInfo, err := client.InfoSync()
if err != nil {
return err
}
rsp := newResponse(types.Result{}, string(resInfo.Data), false)
2017-01-11 22:03:51 -08:00
printResponse(c, rsp)
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)")
}
2016-05-14 00:09:47 -07:00
res := client.SetOptionSync(args[0], args[1])
2017-01-11 22:03:51 -08:00
rsp := newResponse(res, Fmt("%s=%s", args[0], args[1]), false)
printResponse(c, rsp)
return nil
}
2015-11-02 07:39:53 -08:00
// Append a new tx to application
2017-01-12 12:27:08 -08:00
func cmdDeliverTx(c *cli.Context) error {
args := c.Args()
if len(args) != 1 {
2017-01-12 12:27:08 -08:00
return errors.New("Command deliver_tx takes 1 argument")
}
txBytes, err := stringOrHexToBytes(c.Args()[0])
if err != nil {
return err
}
2017-01-12 12:27:08 -08:00
res := client.DeliverTxSync(txBytes)
2017-01-11 22:03:51 -08:00
rsp := newResponse(res, string(res.Data), true)
printResponse(c, rsp)
return nil
2015-11-02 07:39:53 -08:00
}
// 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
}
2016-05-14 00:09:47 -07:00
res := client.CheckTxSync(txBytes)
2017-01-11 22:03:51 -08:00
rsp := newResponse(res, string(res.Data), true)
printResponse(c, rsp)
return nil
2015-11-02 07:39:53 -08:00
}
// Get application Merkle root hash
func cmdCommit(c *cli.Context) error {
2016-05-14 00:09:47 -07:00
res := client.CommitSync()
2017-01-11 22:03:51 -08:00
rsp := newResponse(res, Fmt("0x%X", res.Data), false)
printResponse(c, rsp)
return nil
2015-11-02 07:39:53 -08:00
}
2016-01-22 15:50:11 -08:00
// Query application state
func cmdQuery(c *cli.Context) error {
2016-01-22 15:50:11 -08:00
args := c.Args()
if len(args) != 1 {
return errors.New("Command query takes 1 argument")
2016-01-22 15:50:11 -08:00
}
queryBytes, err := stringOrHexToBytes(c.Args()[0])
if err != nil {
return err
}
2016-05-14 00:09:47 -07:00
res := client.QuerySync(queryBytes)
2017-01-11 22:03:51 -08:00
rsp := newResponse(res, string(res.Data), true)
printResponse(c, rsp)
return nil
2016-01-22 15:50:11 -08:00
}
2015-11-02 07:39:53 -08:00
//--------------------------------------------------------------------------------
2017-01-11 22:03:51 -08:00
func printResponse(c *cli.Context, rsp *response) {
verbose := c.GlobalBool("verbose")
if verbose {
fmt.Println(">", c.Command.Name, strings.Join(c.Args(), " "))
}
2017-01-11 22:03:51 -08:00
if rsp.PrintCode {
fmt.Printf("-> code: %s\n", rsp.Code)
2016-02-07 19:59:19 -08:00
}
2017-01-11 22:03:51 -08:00
//if pr.res.Error != "" {
// fmt.Printf("-> error: %s\n", pr.res.Error)
//}
if rsp.Data != "" {
fmt.Printf("-> data: %s\n", rsp.Data)
2016-02-07 19:59:19 -08:00
}
2017-01-11 22:03:51 -08:00
if rsp.Res.Log != "" {
fmt.Printf("-> log: %s\n", rsp.Res.Log)
2016-02-08 13:47:47 -08:00
}
2017-01-11 22:03:51 -08:00
if verbose {
fmt.Println("")
}
2016-02-07 19:59:19 -08:00
}
2016-07-01 17:22:58 -07:00
// 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" {
2016-07-01 17:22:58 -07:00
b, err := hex.DecodeString(s[2:])
if err != nil {
err = fmt.Errorf("Error decoding hex argument: %s", err.Error())
return nil, err
2016-07-01 17:22:58 -07:00
}
return b, nil
2016-07-01 17:22:58 -07:00
}
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
2016-07-01 17:22:58 -07:00
}