package tx import ( "fmt" "net/http" "strings" "github.com/gorilla/mux" "github.com/spf13/cobra" "github.com/tendermint/tendermint/types" "github.com/spf13/viper" "github.com/cosmos/cosmos-sdk/client/context" "github.com/cosmos/cosmos-sdk/client/flags" "github.com/cosmos/cosmos-sdk/codec" sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/types/rest" ) const ( flagTags = "tags" flagPage = "page" flagLimit = "limit" ) // ---------------------------------------------------------------------------- // CLI // ---------------------------------------------------------------------------- // SearchTxCmd returns a command to search through tagged transactions. func SearchTxCmd(cdc *codec.Codec) *cobra.Command { cmd := &cobra.Command{ Use: "txs", Short: "Search for paginated transactions that match a set of tags", Long: strings.TrimSpace(` Search for transactions that match the exact given tags where results are paginated. Example: $ query txs --tags ':&:' --page 1 --limit 30 `), RunE: func(cmd *cobra.Command, args []string) error { tagsStr := viper.GetString(flagTags) tagsStr = strings.Trim(tagsStr, "'") var tags []string if strings.Contains(tagsStr, "&") { tags = strings.Split(tagsStr, "&") } else { tags = append(tags, tagsStr) } var tmTags []string for _, tag := range tags { if !strings.Contains(tag, ":") { return fmt.Errorf("%s should be of the format :", tagsStr) } else if strings.Count(tag, ":") > 1 { return fmt.Errorf("%s should only contain one : pair", tagsStr) } keyValue := strings.Split(tag, ":") if keyValue[0] == types.TxHeightKey { tag = fmt.Sprintf("%s=%s", keyValue[0], keyValue[1]) } else { tag = fmt.Sprintf("%s='%s'", keyValue[0], keyValue[1]) } tmTags = append(tmTags, tag) } page := viper.GetInt(flagPage) limit := viper.GetInt(flagLimit) cliCtx := context.NewCLIContext().WithCodec(cdc) txs, err := SearchTxs(cliCtx, cdc, tmTags, page, limit) if err != nil { return err } var output []byte if cliCtx.Indent { output, err = cdc.MarshalJSONIndent(txs, "", " ") } else { output, err = cdc.MarshalJSON(txs) } if err != nil { return err } fmt.Println(string(output)) return nil }, } cmd.Flags().StringP(flags.FlagNode, "n", "tcp://localhost:26657", "Node to connect to") viper.BindPFlag(flags.FlagNode, cmd.Flags().Lookup(flags.FlagNode)) cmd.Flags().Bool(flags.FlagTrustNode, false, "Trust connected full node (don't verify proofs for responses)") viper.BindPFlag(flags.FlagTrustNode, cmd.Flags().Lookup(flags.FlagTrustNode)) cmd.Flags().String(flagTags, "", "tag:value list of tags that must match") cmd.Flags().Uint32(flagPage, rest.DefaultPage, "Query a specific page of paginated results") cmd.Flags().Uint32(flagLimit, rest.DefaultLimit, "Query number of transactions results per page returned") cmd.MarkFlagRequired(flagTags) return cmd } // QueryTxCmd implements the default command for a tx query. func QueryTxCmd(cdc *codec.Codec) *cobra.Command { cmd := &cobra.Command{ Use: "tx [hash]", Short: "Find a transaction by hash in a committed block.", Args: cobra.ExactArgs(1), RunE: func(cmd *cobra.Command, args []string) error { cliCtx := context.NewCLIContext().WithCodec(cdc) output, err := queryTx(cdc, cliCtx, args[0]) if err != nil { return err } if output.Empty() { return fmt.Errorf("No transaction found with hash %s", args[0]) } return cliCtx.PrintOutput(output) }, } cmd.Flags().StringP(flags.FlagNode, "n", "tcp://localhost:26657", "Node to connect to") viper.BindPFlag(flags.FlagNode, cmd.Flags().Lookup(flags.FlagNode)) cmd.Flags().Bool(flags.FlagTrustNode, false, "Trust connected full node (don't verify proofs for responses)") viper.BindPFlag(flags.FlagTrustNode, cmd.Flags().Lookup(flags.FlagTrustNode)) return cmd } // ---------------------------------------------------------------------------- // REST // ---------------------------------------------------------------------------- // QueryTxsByTagsRequestHandlerFn implements a REST handler that searches for // transactions by tags. func QueryTxsByTagsRequestHandlerFn(cliCtx context.CLIContext, cdc *codec.Codec) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { var ( tags []string txs []sdk.TxResponse page, limit int ) err := r.ParseForm() if err != nil { rest.WriteErrorResponse(w, http.StatusBadRequest, sdk.AppendMsgToErr("could not parse query parameters", err.Error())) return } if len(r.Form) == 0 { rest.PostProcessResponse(w, cdc, txs, cliCtx.Indent) return } tags, page, limit, err = rest.ParseHTTPArgs(r) if err != nil { rest.WriteErrorResponse(w, http.StatusBadRequest, err.Error()) return } searchResult, err := SearchTxs(cliCtx, cdc, tags, page, limit) if err != nil { rest.WriteErrorResponse(w, http.StatusInternalServerError, err.Error()) return } rest.PostProcessResponse(w, cdc, searchResult, cliCtx.Indent) } } // QueryTxRequestHandlerFn implements a REST handler that queries a transaction // by hash in a committed block. func QueryTxRequestHandlerFn(cdc *codec.Codec, cliCtx context.CLIContext) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { vars := mux.Vars(r) hashHexStr := vars["hash"] output, err := queryTx(cdc, cliCtx, hashHexStr) if err != nil { if strings.Contains(err.Error(), hashHexStr) { rest.WriteErrorResponse(w, http.StatusNotFound, err.Error()) return } rest.WriteErrorResponse(w, http.StatusInternalServerError, err.Error()) return } if output.Empty() { rest.WriteErrorResponse(w, http.StatusNotFound, fmt.Sprintf("no transaction found with hash %s", hashHexStr)) } rest.PostProcessResponse(w, cdc, output, cliCtx.Indent) } }