cosmos-sdk/x/gov/client/cli/tx.go

293 lines
8.3 KiB
Go
Raw Normal View History

2018-06-21 17:19:14 -07:00
package cli
import (
"fmt"
"os"
"strconv"
2018-06-21 17:19:14 -07:00
2018-12-10 06:27:25 -08:00
"github.com/pkg/errors"
2018-06-21 17:19:14 -07:00
"github.com/cosmos/cosmos-sdk/client/context"
2018-08-06 11:11:30 -07:00
"github.com/cosmos/cosmos-sdk/client/utils"
"github.com/cosmos/cosmos-sdk/codec"
2018-06-21 17:19:14 -07:00
sdk "github.com/cosmos/cosmos-sdk/types"
authtxb "github.com/cosmos/cosmos-sdk/x/auth/client/txbuilder"
2018-06-21 17:19:14 -07:00
"github.com/cosmos/cosmos-sdk/x/gov"
2018-08-06 11:11:30 -07:00
2018-08-22 08:56:13 -07:00
"encoding/json"
"io/ioutil"
"strings"
2018-08-22 18:44:35 -07:00
"github.com/spf13/cobra"
"github.com/spf13/viper"
2018-12-10 06:27:25 -08:00
govClientUtils "github.com/cosmos/cosmos-sdk/x/gov/client/utils"
2018-06-21 17:19:14 -07:00
)
const (
flagTitle = "title"
flagDescription = "description"
flagProposalType = "type"
flagDeposit = "deposit"
flagVoter = "voter"
flagOption = "option"
flagDepositor = "depositor"
flagStatus = "status"
flagNumLimit = "limit"
flagProposal = "proposal"
2018-06-21 17:19:14 -07:00
)
type proposal struct {
Title string
Description string
Type string
Deposit string
}
var proposalFlags = []string{
flagTitle,
flagDescription,
flagProposalType,
flagDeposit,
}
2018-08-06 11:11:30 -07:00
// GetCmdSubmitProposal implements submitting a proposal transaction command.
func GetCmdSubmitProposal(cdc *codec.Codec) *cobra.Command {
2018-06-21 17:19:14 -07:00
cmd := &cobra.Command{
Use: "submit-proposal",
2018-06-21 17:19:14 -07:00
Short: "Submit a proposal along with an initial deposit",
Long: strings.TrimSpace(`
Submit a proposal along with an initial deposit. Proposal title, description, type and deposit can be given directly or through a proposal JSON file. For example:
$ gaiacli gov submit-proposal --proposal="path/to/proposal.json" --from mykey
where proposal.json contains:
{
"title": "Test Proposal",
"description": "My awesome proposal",
"type": "Text",
"deposit": "10test"
}
is equivalent to
$ gaiacli gov submit-proposal --title="Test Proposal" --description="My awesome proposal" --type="Text" --deposit="10test" --from mykey
`),
2018-06-21 17:19:14 -07:00
RunE: func(cmd *cobra.Command, args []string) error {
proposal, err := parseSubmitProposalFlags()
if err != nil {
return err
}
2018-06-21 17:19:14 -07:00
txBldr := authtxb.NewTxBuilderFromCLI().WithTxEncoder(utils.GetTxEncoder(cdc))
2018-08-06 11:11:30 -07:00
cliCtx := context.NewCLIContext().
WithCodec(cdc).
WithAccountDecoder(cdc)
2018-08-06 11:11:30 -07:00
// Get from address
from := cliCtx.GetFromAddress()
2018-06-21 17:19:14 -07:00
// Pull associated account
account, err := cliCtx.GetAccount(from)
if err != nil {
return err
}
// Find deposit amount
amount, err := sdk.ParseCoins(proposal.Deposit)
2018-06-21 17:19:14 -07:00
if err != nil {
return err
}
// ensure account has enough coins
if !account.GetCoins().IsAllGTE(amount) {
return errors.Errorf("Address %s doesn't have enough coins to pay for this transaction.", from)
}
proposalType, err := gov.ProposalTypeFromString(proposal.Type)
2018-06-21 17:19:14 -07:00
if err != nil {
return err
}
msg := gov.NewMsgSubmitProposal(proposal.Title, proposal.Description, proposalType, from, amount)
2018-06-21 17:19:14 -07:00
err = msg.ValidateBasic()
if err != nil {
return err
}
2018-09-04 08:29:33 -07:00
if cliCtx.GenerateOnly {
return utils.PrintUnsignedStdTx(os.Stdout, txBldr, cliCtx, []sdk.Msg{msg}, false)
2018-09-04 08:29:33 -07:00
}
2018-08-06 11:11:30 -07:00
// Build and sign the transaction, then broadcast to Tendermint
// proposalID must be returned, and it is a part of response.
cliCtx.PrintResponse = true
return utils.CompleteAndBroadcastTxCLI(txBldr, cliCtx, []sdk.Msg{msg})
2018-06-21 17:19:14 -07:00
},
}
cmd.Flags().String(flagTitle, "", "title of proposal")
cmd.Flags().String(flagDescription, "", "description of proposal")
cmd.Flags().String(flagProposalType, "", "proposalType of proposal, types: text/parameter_change/software_upgrade")
2018-06-21 17:19:14 -07:00
cmd.Flags().String(flagDeposit, "", "deposit of proposal")
cmd.Flags().String(flagProposal, "", "proposal file path (if this path is given, other proposal flags are ignored)")
2018-06-21 17:19:14 -07:00
return cmd
}
func parseSubmitProposalFlags() (*proposal, error) {
proposal := &proposal{}
proposalFile := viper.GetString(flagProposal)
if proposalFile == "" {
proposal.Title = viper.GetString(flagTitle)
proposal.Description = viper.GetString(flagDescription)
proposal.Type = govClientUtils.NormalizeProposalType(viper.GetString(flagProposalType))
proposal.Deposit = viper.GetString(flagDeposit)
return proposal, nil
}
for _, flag := range proposalFlags {
if viper.GetString(flag) != "" {
return nil, fmt.Errorf("--%s flag provided alongside --proposal, which is a noop", flag)
}
}
contents, err := ioutil.ReadFile(proposalFile)
if err != nil {
return nil, err
}
err = json.Unmarshal(contents, proposal)
if err != nil {
return nil, err
}
return proposal, nil
}
2018-08-06 11:11:30 -07:00
// GetCmdDeposit implements depositing tokens for an active proposal.
func GetCmdDeposit(queryRoute string, cdc *codec.Codec) *cobra.Command {
2018-06-21 17:19:14 -07:00
cmd := &cobra.Command{
Use: "deposit [proposal-id] [deposit]",
Args: cobra.ExactArgs(2),
Short: "Deposit tokens for activing proposal",
Long: strings.TrimSpace(`
Submit a deposit for an acive proposal. You can find the proposal-id by running gaiacli query gov proposals:
$ gaiacli tx gov deposit 1 10stake --from mykey
`),
2018-06-21 17:19:14 -07:00
RunE: func(cmd *cobra.Command, args []string) error {
txBldr := authtxb.NewTxBuilderFromCLI().WithTxEncoder(utils.GetTxEncoder(cdc))
2018-08-06 11:11:30 -07:00
cliCtx := context.NewCLIContext().
WithCodec(cdc).
WithAccountDecoder(cdc)
// validate that the proposal id is a uint
proposalID, err := strconv.ParseUint(args[0], 10, 64)
if err != nil {
return fmt.Errorf("proposal-id %s not a valid uint, please input a valid proposal-id", args[0])
}
// check to see if the proposal is in the store
_, err = govClientUtils.QueryProposalByID(proposalID, cliCtx, cdc, queryRoute)
if err != nil {
return fmt.Errorf("Failed to fetch proposal-id %d: %s", proposalID, err)
}
from := cliCtx.GetFromAddress()
2018-06-21 17:19:14 -07:00
// Fetch associated account
account, err := cliCtx.GetAccount(from)
if err != nil {
return err
}
2018-06-21 17:19:14 -07:00
// Get amount of coins
amount, err := sdk.ParseCoins(args[1])
2018-06-21 17:19:14 -07:00
if err != nil {
return err
}
// ensure account has enough coins
if !account.GetCoins().IsAllGTE(amount) {
return errors.Errorf("Address %s doesn't have enough coins to pay for this transaction.", from)
}
msg := gov.NewMsgDeposit(from, proposalID, amount)
2018-06-21 17:19:14 -07:00
err = msg.ValidateBasic()
if err != nil {
return err
}
2018-09-04 08:29:33 -07:00
if cliCtx.GenerateOnly {
return utils.PrintUnsignedStdTx(os.Stdout, txBldr, cliCtx, []sdk.Msg{msg}, false)
2018-09-04 08:29:33 -07:00
}
// Build and sign the transaction, then broadcast to a Tendermint node.
return utils.CompleteAndBroadcastTxCLI(txBldr, cliCtx, []sdk.Msg{msg})
2018-06-21 17:19:14 -07:00
},
}
return cmd
}
2018-08-06 11:11:30 -07:00
// GetCmdVote implements creating a new vote command.
func GetCmdVote(queryRoute string, cdc *codec.Codec) *cobra.Command {
2018-06-21 17:19:14 -07:00
cmd := &cobra.Command{
Use: "vote [proposal-id] [option]",
Args: cobra.ExactArgs(2),
Short: "Vote for an active proposal, options: yes/no/no_with_veto/abstain",
Long: strings.TrimSpace(`
Submit a vote for an acive proposal. You can find the proposal-id by running gaiacli query gov proposals:
$ gaiacli tx gov vote 1 yes --from mykey
`),
2018-06-21 17:19:14 -07:00
RunE: func(cmd *cobra.Command, args []string) error {
txBldr := authtxb.NewTxBuilderFromCLI().WithTxEncoder(utils.GetTxEncoder(cdc))
2018-08-06 11:11:30 -07:00
cliCtx := context.NewCLIContext().
WithCodec(cdc).
WithAccountDecoder(cdc)
2018-06-21 17:19:14 -07:00
// Get voting address
from := cliCtx.GetFromAddress()
2018-06-21 17:19:14 -07:00
// validate that the proposal id is a uint
proposalID, err := strconv.ParseUint(args[0], 10, 64)
if err != nil {
return fmt.Errorf("proposal-id %s not a valid int, please input a valid proposal-id", args[0])
}
2018-06-21 17:19:14 -07:00
// check to see if the proposal is in the store
_, err = govClientUtils.QueryProposalByID(proposalID, cliCtx, cdc, queryRoute)
if err != nil {
return fmt.Errorf("Failed to fetch proposal-id %d: %s", proposalID, err)
}
// Find out which vote option user chose
byteVoteOption, err := gov.VoteOptionFromString(govClientUtils.NormalizeVoteOption(args[1]))
2018-06-21 17:19:14 -07:00
if err != nil {
return err
}
// Build vote message and run basic validation
msg := gov.NewMsgVote(from, proposalID, byteVoteOption)
2018-06-21 17:19:14 -07:00
err = msg.ValidateBasic()
if err != nil {
return err
}
// If generate only print the transaction
2018-09-04 08:29:33 -07:00
if cliCtx.GenerateOnly {
return utils.PrintUnsignedStdTx(os.Stdout, txBldr, cliCtx, []sdk.Msg{msg}, false)
2018-09-04 08:29:33 -07:00
}
// Build and sign the transaction, then broadcast to a Tendermint node.
return utils.CompleteAndBroadcastTxCLI(txBldr, cliCtx, []sdk.Msg{msg})
2018-06-21 17:19:14 -07:00
},
}
return cmd
}