0g-chain/internal/x/paychan/client/cmd/cmd.go

302 lines
8.3 KiB
Go
Raw Normal View History

2018-07-08 22:09:07 +00:00
package cli
import (
2018-07-14 22:27:56 +00:00
"encoding/base64"
2018-07-08 22:09:07 +00:00
"fmt"
2018-07-14 22:27:56 +00:00
"github.com/pkg/errors"
"github.com/spf13/cobra"
"github.com/spf13/viper"
2018-07-14 00:09:02 +00:00
"github.com/cosmos/cosmos-sdk/client/context"
2018-07-14 22:27:56 +00:00
"github.com/cosmos/cosmos-sdk/client/keys"
2018-07-14 00:09:02 +00:00
sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/cosmos/cosmos-sdk/wire"
2018-07-14 22:27:56 +00:00
"github.com/cosmos/cosmos-sdk/x/auth"
2018-07-15 14:08:08 +00:00
authcmd "github.com/cosmos/cosmos-sdk/x/auth/client/cli"
2018-07-14 22:27:56 +00:00
2018-07-14 00:09:02 +00:00
"github.com/kava-labs/kava/internal/x/paychan"
2018-07-08 22:09:07 +00:00
)
// list of functions that return pointers to cobra commands
// No local storage needed for cli acting as a sender
2018-07-14 22:27:56 +00:00
// Currently minimum set of cli commands are implemented:
// create paychan - create and fund
// generate new paychan state - print a half signed close tx (sender signs)
// close paychan - close using state (receiver signs)
// Future cli commands:
2018-07-08 22:09:07 +00:00
// create paychan
// close paychan
// get paychan(s)
// send paychan payment
// get balance from receiver
2018-08-24 23:18:41 +00:00
/*
2018-07-14 00:09:02 +00:00
func CreatePaychanCmd(cdc *wire.Codec) *cobra.Command {
flagTo := "to"
flagAmount := "amount"
cmd := &cobra.Command{
Use: "create",
Short: "Create a new payment channel",
Long: "Create a new payment channel from a local address to a remote address, funded with some amount of coins. These coins are removed from the sender account and put into the payment channel.",
Args: cobra.NoArgs,
2018-07-08 22:09:07 +00:00
RunE: func(cmd *cobra.Command, args []string) error {
2018-07-14 22:27:56 +00:00
// Create a "client context" stuct populated with info from common flags
2018-07-15 14:08:08 +00:00
ctx := context.NewCoreContextFromViper().WithDecoder(authcmd.GetAccountDecoder(cdc))
2018-07-08 22:09:07 +00:00
2018-07-14 00:09:02 +00:00
// Get sender adress
senderAddress, err := ctx.GetFromAddress()
if err != nil {
return err
}
2018-07-08 22:09:07 +00:00
2018-07-14 00:09:02 +00:00
// Get receiver address
toStr := viper.GetString(flagTo)
receiverAddress, err := sdk.GetAccAddressBech32(toStr)
2018-07-08 22:09:07 +00:00
if err != nil {
return err
}
2018-07-14 00:09:02 +00:00
// Get channel funding amount
amountString := viper.GetString(flagAmount)
amount, err := sdk.ParseCoins(amountString)
2018-07-08 22:09:07 +00:00
if err != nil {
return err
}
2018-07-14 00:09:02 +00:00
// Create the create channel msg to send
// TODO write NewMsgCreate func?
msg := paychan.MsgCreate{
2018-07-14 22:27:56 +00:00
Sender: senderAddress,
Receiver: receiverAddress,
Amount: amount,
2018-07-08 22:09:07 +00:00
}
2018-07-14 22:27:56 +00:00
// Build and sign the transaction, then broadcast to the blockchain
2018-07-14 00:09:02 +00:00
res, err := ctx.EnsureSignBuildBroadcast(ctx.FromAddressName, msg, cdc)
if err != nil {
return err
}
fmt.Printf("Committed at block %d. Hash: %s\n", res.Height, res.Hash.String())
2018-07-14 22:27:56 +00:00
return nil
2018-07-14 00:09:02 +00:00
},
}
cmd.Flags().String(flagTo, "", "Recipient address of the payment channel")
cmd.Flags().String(flagAmount, "", "Amount of coins to fund the paymetn channel with")
return cmd
}
2018-07-08 22:09:07 +00:00
2018-07-14 22:27:56 +00:00
func GenerateNewStateCmd(cdc *wire.Codec) *cobra.Command {
2018-07-14 00:09:02 +00:00
flagId := "id"
flagTo := "to"
flagAmount := "amount"
cmd := &cobra.Command{
2018-07-14 22:27:56 +00:00
Use: "new-state",
Short: "Generate a new payment channel state.",
2018-07-15 14:08:08 +00:00
Long: "Generate a new state for an existing payment channel and print it out. The new state is represented as a half signed close transaction, signed by the sender.",
2018-07-14 00:09:02 +00:00
Args: cobra.NoArgs,
RunE: func(cmd *cobra.Command, args []string) error {
2018-07-14 22:27:56 +00:00
// Create a "client context" stuct populated with info from common flags
2018-07-15 14:08:08 +00:00
ctx := context.NewCoreContextFromViper().WithDecoder(authcmd.GetAccountDecoder(cdc))
2018-07-14 22:27:56 +00:00
2018-07-14 00:09:02 +00:00
// Get sender adress
senderAddress, err := ctx.GetFromAddress()
2018-07-08 22:09:07 +00:00
if err != nil {
return err
}
2018-07-14 22:27:56 +00:00
// Get the paychan id
id := viper.GetInt64(flagId)
2018-07-14 00:09:02 +00:00
// Get receiver address
toStr := viper.GetString(flagTo)
receiverAddress, err := sdk.GetAccAddressBech32(toStr)
if err != nil {
return err
}
2018-07-14 22:27:56 +00:00
2018-07-14 00:09:02 +00:00
// Get channel receiver amount
amountString := viper.GetString(flagAmount)
amount, err := sdk.ParseCoins(amountString)
if err != nil {
return err
}
2018-07-14 22:27:56 +00:00
// create close paychan msg
2018-07-14 00:09:02 +00:00
msg := paychan.MsgClose{
2018-07-14 22:27:56 +00:00
Sender: senderAddress,
Receiver: receiverAddress,
Id: id,
ReceiverAmount: amount,
2018-07-14 00:09:02 +00:00
}
2018-07-14 22:27:56 +00:00
// Sign the msg as the sender
2018-07-14 00:09:02 +00:00
txBytes, err := EnsureSignBuild(ctx, ctx.FromAddressName, msg, cdc)
2018-07-08 22:09:07 +00:00
if err != nil {
return err
}
2018-07-14 00:09:02 +00:00
2018-07-14 22:27:56 +00:00
// Print out the signed msg
fmt.Println("txBytes:", txBytes)
2018-07-15 14:08:08 +00:00
//encodedTxBytes := make([]byte, base64.StdEncoding.EncodedLen(len(txBytes)))
encodedTxBytes := base64.StdEncoding.EncodeToString(txBytes)
2018-07-14 22:27:56 +00:00
fmt.Println("base64TxBytes:", encodedTxBytes)
return nil
2018-07-14 00:09:02 +00:00
},
}
cmd.Flags().Int(flagId, 0, "ID of the payment channel.")
cmd.Flags().String(flagTo, "", "Recipient address of the payment channel")
cmd.Flags().String(flagAmount, "", "Amount of coins to fund the paymetn channel with")
return cmd
}
2018-07-14 22:27:56 +00:00
func ClosePaychanCmd(cdc *wire.Codec) *cobra.Command {
flagState := "state"
2018-07-14 00:09:02 +00:00
2018-07-14 22:27:56 +00:00
cmd := &cobra.Command{
Use: "close",
Short: "Close a payment channel, given a state",
2018-07-15 14:08:08 +00:00
Long: "Close an existing payment channel with a state received from a sender. This signs it as the receiver before submitting to the blockchain.",
2018-07-14 22:27:56 +00:00
Args: cobra.NoArgs,
RunE: func(cmd *cobra.Command, args []string) error {
2018-07-15 14:08:08 +00:00
ctx := context.NewCoreContextFromViper().WithDecoder(authcmd.GetAccountDecoder(cdc))
2018-07-14 22:27:56 +00:00
// Get the sender-signed close tx
state := viper.GetString(flagState)
txBytes, err := base64.StdEncoding.DecodeString(state)
if err != nil {
return err
}
stdTx := auth.StdTx{}
2018-07-15 14:08:08 +00:00
cdc.UnmarshalBinary(txBytes, &stdTx)
2018-07-14 22:27:56 +00:00
// Sign close tx
// ensure contxt has up to date account and sequence numbers
ctx, err = Ensure(ctx)
if err != nil {
return err
}
// Sign message (asks user for password)
_, sig, err := UserSignMsg(ctx, ctx.FromAddressName, stdTx.Msg)
if err != nil {
return err
}
// Append signature to close tx
stdTx.Signatures = append(stdTx.Signatures, sig)
// encode close tx
txBytes, err = cdc.MarshalBinary(stdTx)
if err != nil {
return err
}
// Broadcast close tx to the blockchain
res, err := ctx.BroadcastTx(txBytes)
if err != nil {
return err
}
fmt.Printf("Committed at block %d. Hash: %s\n", res.Height, res.Hash.String())
return nil
},
}
cmd.Flags().String(flagState, "", "State received from sender.")
return cmd
}
// HELPER FUNCTIONS
// This is a partial refactor of cosmos-sdk/client/context.
// Existing API was awkard to use for paychans.
func Ensure(ctx context.CoreContext) (context.CoreContext, error) {
ctx, err := context.EnsureAccountNumber(ctx)
2018-07-14 00:09:02 +00:00
if err != nil {
2018-07-14 22:27:56 +00:00
return ctx, err
2018-07-14 00:09:02 +00:00
}
// default to next sequence number if none provided
2018-07-14 22:27:56 +00:00
ctx, err = context.EnsureSequence(ctx)
2018-07-14 00:09:02 +00:00
if err != nil {
2018-07-14 22:27:56 +00:00
return ctx, err
2018-07-14 00:09:02 +00:00
}
2018-07-14 22:27:56 +00:00
return ctx, nil
}
func UserSignMsg(ctx context.CoreContext, name string, msg sdk.Msg) (signMsg auth.StdSignMsg, stdSig auth.StdSignature, err error) {
// TODO check how to handle non error return values on error. Returning empty versions doesn't seem right.
2018-07-14 00:09:02 +00:00
passphrase, err := ctx.GetPassphraseFromStdin(name)
if err != nil {
2018-07-14 22:27:56 +00:00
return signMsg, stdSig, err
2018-07-14 00:09:02 +00:00
}
2018-07-14 22:27:56 +00:00
// build the Sign Messsage from the Standard Message
chainID := ctx.ChainID
if chainID == "" {
return signMsg, stdSig, errors.Errorf("Chain ID required but not specified")
}
accnum := ctx.AccountNumber
sequence := ctx.Sequence
signMsg = auth.StdSignMsg{
ChainID: chainID,
AccountNumbers: []int64{accnum},
Sequences: []int64{sequence},
Msg: msg,
Fee: auth.NewStdFee(ctx.Gas, sdk.Coin{}), // TODO run simulate to estimate gas?
}
keybase, err := keys.GetKeyBase()
2018-07-14 00:09:02 +00:00
if err != nil {
2018-07-14 22:27:56 +00:00
return signMsg, stdSig, err
}
// sign and build
bz := signMsg.Bytes()
sig, pubkey, err := keybase.Sign(name, passphrase, bz)
if err != nil {
return signMsg, stdSig, err
}
stdSig = auth.StdSignature{
PubKey: pubkey,
Signature: sig,
AccountNumber: accnum,
Sequence: sequence,
2018-07-14 00:09:02 +00:00
}
2018-07-14 22:27:56 +00:00
return signMsg, stdSig, nil
2018-07-14 00:09:02 +00:00
}
2018-07-14 22:27:56 +00:00
func Build(cdc *wire.Codec, signMsg auth.StdSignMsg, sig auth.StdSignature) ([]byte, error) {
tx := auth.NewStdTx(signMsg.Msg, signMsg.Fee, []auth.StdSignature{sig})
return cdc.MarshalBinary(tx)
}
2018-07-14 00:09:02 +00:00
2018-07-14 22:27:56 +00:00
func EnsureSignBuild(ctx context.CoreContext, name string, msg sdk.Msg, cdc *wire.Codec) ([]byte, error) {
//Ensure context has up to date account and sequence numbers
ctx, err := Ensure(ctx)
if err != nil {
return nil, err
2018-07-08 22:09:07 +00:00
}
2018-07-14 22:27:56 +00:00
// Sign message (asks user for password)
signMsg, sig, err := UserSignMsg(ctx, name, msg)
if err != nil {
return nil, err
}
// Create tx and marshal
txBytes, err := Build(cdc, signMsg, sig)
if err != nil {
return nil, err
}
return txBytes, nil
2018-07-08 22:09:07 +00:00
}
2018-08-24 23:18:41 +00:00
*/