Merge PR #3428: SendMsg and MultiSendMsg seperation

This commit is contained in:
Sunny Aggarwal 2019-02-04 15:58:02 -08:00 committed by Jack Zampolin
parent 082295172e
commit 18eee0a3a8
13 changed files with 349 additions and 95 deletions

View File

@ -259,7 +259,7 @@ func TestCoinSendAccAuto(t *testing.T) {
require.Equal(t, expectedBalance.Amount.SubRaw(1), coins[0].Amount) require.Equal(t, expectedBalance.Amount.SubRaw(1), coins[0].Amount)
} }
func TestCoinSendGenerateOnly(t *testing.T) { func TestCoinMultiSendGenerateOnly(t *testing.T) {
addr, seed := CreateAddr(t, name1, pw, GetKeyBase(t)) addr, seed := CreateAddr(t, name1, pw, GetKeyBase(t))
cleanup, _, _, port := InitializeTestLCD(t, 1, []sdk.AccAddress{addr}) cleanup, _, _, port := InitializeTestLCD(t, 1, []sdk.AccAddress{addr})
defer cleanup() defer cleanup()
@ -277,7 +277,7 @@ func TestCoinSendGenerateOnly(t *testing.T) {
require.Equal(t, memo, stdTx.Memo) require.Equal(t, memo, stdTx.Memo)
require.NotZero(t, stdTx.Fee.Gas) require.NotZero(t, stdTx.Fee.Gas)
require.IsType(t, stdTx.GetMsgs()[0], bank.MsgSend{}) require.IsType(t, stdTx.GetMsgs()[0], bank.MsgSend{})
require.Equal(t, addr, stdTx.GetMsgs()[0].(bank.MsgSend).Inputs[0].Address) require.Equal(t, addr, stdTx.GetMsgs()[0].(bank.MsgSend).FromAddress)
} }
func TestCoinSendGenerateSignAndBroadcast(t *testing.T) { func TestCoinSendGenerateSignAndBroadcast(t *testing.T) {

View File

@ -23,7 +23,7 @@ func ExampleTxSendSize() {
priv2 := secp256k1.GenPrivKeySecp256k1([]byte{1}) priv2 := secp256k1.GenPrivKeySecp256k1([]byte{1})
addr2 := sdk.AccAddress(priv2.PubKey().Address()) addr2 := sdk.AccAddress(priv2.PubKey().Address())
coins := sdk.Coins{sdk.NewCoin("denom", sdk.NewInt(10))} coins := sdk.Coins{sdk.NewCoin("denom", sdk.NewInt(10))}
msg1 := bank.MsgSend{ msg1 := bank.MsgMultiSend{
Inputs: []bank.Input{bank.NewInput(addr1, coins)}, Inputs: []bank.Input{bank.NewInput(addr1, coins)},
Outputs: []bank.Output{bank.NewOutput(addr2, coins)}, Outputs: []bank.Output{bank.NewOutput(addr2, coins)},
} }

View File

@ -266,7 +266,8 @@ func randIntBetween(r *rand.Rand, min, max int) int {
func testAndRunTxs(app *GaiaApp) []simulation.WeightedOperation { func testAndRunTxs(app *GaiaApp) []simulation.WeightedOperation {
return []simulation.WeightedOperation{ return []simulation.WeightedOperation{
{5, authsim.SimulateDeductFee(app.accountKeeper, app.feeCollectionKeeper)}, {5, authsim.SimulateDeductFee(app.accountKeeper, app.feeCollectionKeeper)},
{100, banksim.SingleInputSendMsg(app.accountKeeper, app.bankKeeper)}, {100, banksim.SendMsg(app.accountKeeper, app.bankKeeper)},
{10, banksim.SingleInputMsgMultiSend(app.accountKeeper, app.bankKeeper)},
{50, distrsim.SimulateMsgSetWithdrawAddress(app.accountKeeper, app.distrKeeper)}, {50, distrsim.SimulateMsgSetWithdrawAddress(app.accountKeeper, app.distrKeeper)},
{50, distrsim.SimulateMsgWithdrawDelegatorReward(app.accountKeeper, app.distrKeeper)}, {50, distrsim.SimulateMsgWithdrawDelegatorReward(app.accountKeeper, app.distrKeeper)},
{50, distrsim.SimulateMsgWithdrawValidatorCommission(app.accountKeeper, app.distrKeeper)}, {50, distrsim.SimulateMsgWithdrawValidatorCommission(app.accountKeeper, app.distrKeeper)},

View File

@ -45,18 +45,20 @@ var (
manyCoins = sdk.Coins{sdk.NewInt64Coin("foocoin", 1), sdk.NewInt64Coin("barcoin", 1)} manyCoins = sdk.Coins{sdk.NewInt64Coin("foocoin", 1), sdk.NewInt64Coin("barcoin", 1)}
freeFee = auth.NewStdFee(100000, sdk.Coins{sdk.NewInt64Coin("foocoin", 0)}) freeFee = auth.NewStdFee(100000, sdk.Coins{sdk.NewInt64Coin("foocoin", 0)})
sendMsg1 = MsgSend{ sendMsg1 = NewMsgSend(addr1, addr2, coins)
multiSendMsg1 = MsgMultiSend{
Inputs: []Input{NewInput(addr1, coins)}, Inputs: []Input{NewInput(addr1, coins)},
Outputs: []Output{NewOutput(addr2, coins)}, Outputs: []Output{NewOutput(addr2, coins)},
} }
sendMsg2 = MsgSend{ multiSendMsg2 = MsgMultiSend{
Inputs: []Input{NewInput(addr1, coins)}, Inputs: []Input{NewInput(addr1, coins)},
Outputs: []Output{ Outputs: []Output{
NewOutput(addr2, halfCoins), NewOutput(addr2, halfCoins),
NewOutput(addr3, halfCoins), NewOutput(addr3, halfCoins),
}, },
} }
sendMsg3 = MsgSend{ multiSendMsg3 = MsgMultiSend{
Inputs: []Input{ Inputs: []Input{
NewInput(addr1, coins), NewInput(addr1, coins),
NewInput(addr4, coins), NewInput(addr4, coins),
@ -66,7 +68,7 @@ var (
NewOutput(addr3, coins), NewOutput(addr3, coins),
}, },
} }
sendMsg4 = MsgSend{ multiSendMsg4 = MsgMultiSend{
Inputs: []Input{ Inputs: []Input{
NewInput(addr2, coins), NewInput(addr2, coins),
}, },
@ -74,7 +76,7 @@ var (
NewOutput(addr1, coins), NewOutput(addr1, coins),
}, },
} }
sendMsg5 = MsgSend{ multiSendMsg5 = MsgMultiSend{
Inputs: []Input{ Inputs: []Input{
NewInput(addr1, manyCoins), NewInput(addr1, manyCoins),
}, },
@ -102,7 +104,7 @@ func getInitChainer(mapp *mock.App, keeper BaseKeeper) sdk.InitChainer {
} }
} }
func TestMsgSendWithAccounts(t *testing.T) { func TestMsgMultiSendWithAccounts(t *testing.T) {
mapp := getMockApp(t) mapp := getMockApp(t)
acc := &auth.BaseAccount{ acc := &auth.BaseAccount{
Address: addr1, Address: addr1,
@ -119,7 +121,7 @@ func TestMsgSendWithAccounts(t *testing.T) {
testCases := []appTestCase{ testCases := []appTestCase{
{ {
msgs: []sdk.Msg{sendMsg1}, msgs: []sdk.Msg{multiSendMsg1},
accNums: []uint64{0}, accNums: []uint64{0},
accSeqs: []uint64{0}, accSeqs: []uint64{0},
expSimPass: true, expSimPass: true,
@ -131,7 +133,7 @@ func TestMsgSendWithAccounts(t *testing.T) {
}, },
}, },
{ {
msgs: []sdk.Msg{sendMsg1, sendMsg2}, msgs: []sdk.Msg{multiSendMsg1, multiSendMsg2},
accNums: []uint64{0}, accNums: []uint64{0},
accSeqs: []uint64{0}, accSeqs: []uint64{0},
expSimPass: true, // doesn't check signature expSimPass: true, // doesn't check signature
@ -149,7 +151,7 @@ func TestMsgSendWithAccounts(t *testing.T) {
} }
} }
func TestMsgSendMultipleOut(t *testing.T) { func TestMsgMultiSendMultipleOut(t *testing.T) {
mapp := getMockApp(t) mapp := getMockApp(t)
acc1 := &auth.BaseAccount{ acc1 := &auth.BaseAccount{
@ -165,7 +167,7 @@ func TestMsgSendMultipleOut(t *testing.T) {
testCases := []appTestCase{ testCases := []appTestCase{
{ {
msgs: []sdk.Msg{sendMsg2}, msgs: []sdk.Msg{multiSendMsg2},
accNums: []uint64{0}, accNums: []uint64{0},
accSeqs: []uint64{0}, accSeqs: []uint64{0},
expSimPass: true, expSimPass: true,
@ -188,7 +190,7 @@ func TestMsgSendMultipleOut(t *testing.T) {
} }
} }
func TestSengMsgMultipleInOut(t *testing.T) { func TestMsgMultiSendMultipleInOut(t *testing.T) {
mapp := getMockApp(t) mapp := getMockApp(t)
acc1 := &auth.BaseAccount{ acc1 := &auth.BaseAccount{
@ -208,7 +210,7 @@ func TestSengMsgMultipleInOut(t *testing.T) {
testCases := []appTestCase{ testCases := []appTestCase{
{ {
msgs: []sdk.Msg{sendMsg3}, msgs: []sdk.Msg{multiSendMsg3},
accNums: []uint64{0, 0}, accNums: []uint64{0, 0},
accSeqs: []uint64{0, 0}, accSeqs: []uint64{0, 0},
expSimPass: true, expSimPass: true,
@ -232,7 +234,7 @@ func TestSengMsgMultipleInOut(t *testing.T) {
} }
} }
func TestMsgSendDependent(t *testing.T) { func TestMsgMultiSendDependent(t *testing.T) {
mapp := getMockApp(t) mapp := getMockApp(t)
acc1 := &auth.BaseAccount{ acc1 := &auth.BaseAccount{
@ -244,7 +246,7 @@ func TestMsgSendDependent(t *testing.T) {
testCases := []appTestCase{ testCases := []appTestCase{
{ {
msgs: []sdk.Msg{sendMsg1}, msgs: []sdk.Msg{multiSendMsg1},
accNums: []uint64{0}, accNums: []uint64{0},
accSeqs: []uint64{0}, accSeqs: []uint64{0},
expSimPass: true, expSimPass: true,
@ -256,7 +258,7 @@ func TestMsgSendDependent(t *testing.T) {
}, },
}, },
{ {
msgs: []sdk.Msg{sendMsg4}, msgs: []sdk.Msg{multiSendMsg4},
accNums: []uint64{0}, accNums: []uint64{0},
accSeqs: []uint64{0}, accSeqs: []uint64{0},
expSimPass: true, expSimPass: true,

View File

@ -57,3 +57,33 @@ func BenchmarkOneBankSendTxPerBlock(b *testing.B) {
benchmarkApp.Commit() benchmarkApp.Commit()
} }
} }
func BenchmarkOneBankMultiSendTxPerBlock(b *testing.B) {
benchmarkApp, _ := getBenchmarkMockApp()
// Add an account at genesis
acc := &auth.BaseAccount{
Address: addr1,
// Some value conceivably higher than the benchmarks would ever go
Coins: sdk.Coins{sdk.NewInt64Coin("foocoin", 100000000000)},
}
accs := []auth.Account{acc}
// Construct genesis state
mock.SetGenesis(benchmarkApp, accs)
// Precompute all txs
txs := mock.GenSequenceOfTxs([]sdk.Msg{multiSendMsg1}, []uint64{0}, []uint64{uint64(0)}, b.N, priv1)
b.ResetTimer()
// Run this with a profiler, so its easy to distinguish what time comes from
// Committing, and what time comes from Check/Deliver Tx.
for i := 0; i < b.N; i++ {
benchmarkApp.BeginBlock(abci.RequestBeginBlock{})
x := benchmarkApp.Check(txs[i])
if !x.IsOK() {
panic("something is broken in checking transaction")
}
benchmarkApp.Deliver(txs[i])
benchmarkApp.EndBlock(abci.RequestEndBlock{})
benchmarkApp.Commit()
}
}

View File

@ -9,7 +9,7 @@ import (
"github.com/cosmos/cosmos-sdk/codec" "github.com/cosmos/cosmos-sdk/codec"
sdk "github.com/cosmos/cosmos-sdk/types" sdk "github.com/cosmos/cosmos-sdk/types"
authtxb "github.com/cosmos/cosmos-sdk/x/auth/client/txbuilder" authtxb "github.com/cosmos/cosmos-sdk/x/auth/client/txbuilder"
bankClient "github.com/cosmos/cosmos-sdk/x/bank/client" "github.com/cosmos/cosmos-sdk/x/bank"
"github.com/pkg/errors" "github.com/pkg/errors"
"github.com/spf13/cobra" "github.com/spf13/cobra"
@ -62,7 +62,7 @@ func SendTxCmd(cdc *codec.Codec) *cobra.Command {
} }
// build and sign the transaction, then broadcast to Tendermint // build and sign the transaction, then broadcast to Tendermint
msg := bankClient.CreateMsg(from, to, coins) msg := bank.NewMsgSend(from, to, coins)
if cliCtx.GenerateOnly { if cliCtx.GenerateOnly {
return utils.PrintUnsignedStdTx(os.Stdout, txBldr, cliCtx, []sdk.Msg{msg}, false) return utils.PrintUnsignedStdTx(os.Stdout, txBldr, cliCtx, []sdk.Msg{msg}, false)
} }

View File

@ -10,7 +10,6 @@ import (
"github.com/cosmos/cosmos-sdk/crypto/keys" "github.com/cosmos/cosmos-sdk/crypto/keys"
sdk "github.com/cosmos/cosmos-sdk/types" sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/cosmos/cosmos-sdk/x/bank" "github.com/cosmos/cosmos-sdk/x/bank"
bankclient "github.com/cosmos/cosmos-sdk/x/bank/client"
"github.com/gorilla/mux" "github.com/gorilla/mux"
) )
@ -64,7 +63,7 @@ func SendRequestHandlerFn(cdc *codec.Codec, kb keys.Keybase, cliCtx context.CLIC
return return
} }
msg := bankclient.CreateMsg(fromAddr, toAddr, req.Amount) msg := bank.NewMsgSend(fromAddr, toAddr, req.Amount)
rest.WriteGenerateStdTxResponse(w, cdc, cliCtx, req.BaseReq, []sdk.Msg{msg}) rest.WriteGenerateStdTxResponse(w, cdc, cliCtx, req.BaseReq, []sdk.Msg{msg})
return return
} }
@ -77,7 +76,7 @@ func SendRequestHandlerFn(cdc *codec.Codec, kb keys.Keybase, cliCtx context.CLIC
} }
cliCtx = cliCtx.WithFromName(fromName).WithFromAddress(fromAddress) cliCtx = cliCtx.WithFromName(fromName).WithFromAddress(fromAddress)
msg := bankclient.CreateMsg(cliCtx.GetFromAddress(), toAddr, req.Amount) msg := bank.NewMsgSend(cliCtx.GetFromAddress(), toAddr, req.Amount)
rest.CompleteAndBroadcastTxREST(w, r, cliCtx, req.BaseReq, []sdk.Msg{msg}, cdc) rest.CompleteAndBroadcastTxREST(w, r, cliCtx, req.BaseReq, []sdk.Msg{msg}, cdc)
} }

View File

@ -1,14 +0,0 @@
package client
import (
sdk "github.com/cosmos/cosmos-sdk/types"
bank "github.com/cosmos/cosmos-sdk/x/bank"
)
// create the sendTx msg
func CreateMsg(from sdk.AccAddress, to sdk.AccAddress, coins sdk.Coins) sdk.Msg {
input := bank.NewInput(from, coins)
output := bank.NewOutput(to, coins)
msg := bank.NewMsgSend([]bank.Input{input}, []bank.Output{output})
return msg
}

View File

@ -7,6 +7,7 @@ import (
// Register concrete types on codec codec // Register concrete types on codec codec
func RegisterCodec(cdc *codec.Codec) { func RegisterCodec(cdc *codec.Codec) {
cdc.RegisterConcrete(MsgSend{}, "cosmos-sdk/Send", nil) cdc.RegisterConcrete(MsgSend{}, "cosmos-sdk/Send", nil)
cdc.RegisterConcrete(MsgMultiSend{}, "cosmos-sdk/MultiSend", nil)
} }
var msgCdc = codec.New() var msgCdc = codec.New()

View File

@ -10,6 +10,8 @@ func NewHandler(k Keeper) sdk.Handler {
switch msg := msg.(type) { switch msg := msg.(type) {
case MsgSend: case MsgSend:
return handleMsgSend(ctx, k, msg) return handleMsgSend(ctx, k, msg)
case MsgMultiSend:
return handleMsgMultiSend(ctx, k, msg)
default: default:
errMsg := "Unrecognized bank Msg type: %s" + msg.Type() errMsg := "Unrecognized bank Msg type: %s" + msg.Type()
return sdk.ErrUnknownRequest(errMsg).Result() return sdk.ErrUnknownRequest(errMsg).Result()
@ -19,6 +21,21 @@ func NewHandler(k Keeper) sdk.Handler {
// Handle MsgSend. // Handle MsgSend.
func handleMsgSend(ctx sdk.Context, k Keeper, msg MsgSend) sdk.Result { func handleMsgSend(ctx sdk.Context, k Keeper, msg MsgSend) sdk.Result {
if !k.GetSendEnabled(ctx) {
return ErrSendDisabled(k.Codespace()).Result()
}
tags, err := k.SendCoins(ctx, msg.FromAddress, msg.ToAddress, msg.Amount)
if err != nil {
return err.Result()
}
return sdk.Result{
Tags: tags,
}
}
// Handle MsgMultiSend.
func handleMsgMultiSend(ctx sdk.Context, k Keeper, msg MsgMultiSend) sdk.Result {
// NOTE: totalIn == totalOut should already have been checked // NOTE: totalIn == totalOut should already have been checked
if !k.GetSendEnabled(ctx) { if !k.GetSendEnabled(ctx) {
return ErrSendDisabled(k.Codespace()).Result() return ErrSendDisabled(k.Codespace()).Result()

View File

@ -9,15 +9,16 @@ const RouterKey = "bank"
// MsgSend - high level transaction of the coin module // MsgSend - high level transaction of the coin module
type MsgSend struct { type MsgSend struct {
Inputs []Input `json:"inputs"` FromAddress sdk.AccAddress `json:"from_address"`
Outputs []Output `json:"outputs"` ToAddress sdk.AccAddress `json:"to_address"`
Amount sdk.Coins `json:"amount"`
} }
var _ sdk.Msg = MsgSend{} var _ sdk.Msg = MsgSend{}
// NewMsgSend - construct arbitrary multi-in, multi-out send msg. // NewMsgSend - construct arbitrary multi-in, multi-out send msg.
func NewMsgSend(in []Input, out []Output) MsgSend { func NewMsgSend(fromAddr, toAddr sdk.AccAddress, amount sdk.Coins) MsgSend {
return MsgSend{Inputs: in, Outputs: out} return MsgSend{FromAddress: fromAddr, ToAddress: toAddr, Amount: amount}
} }
// Implements Msg. // Implements Msg.
@ -27,6 +28,48 @@ func (msg MsgSend) Type() string { return "send" }
// Implements Msg. // Implements Msg.
func (msg MsgSend) ValidateBasic() sdk.Error { func (msg MsgSend) ValidateBasic() sdk.Error {
if msg.FromAddress.Empty() {
return sdk.ErrInvalidAddress("missing sender address")
}
if msg.ToAddress.Empty() {
return sdk.ErrInvalidAddress("missing recipient address")
}
if !msg.Amount.IsPositive() {
return sdk.ErrInsufficientCoins("send amount must be positive")
}
return nil
}
// Implements Msg.
func (msg MsgSend) GetSignBytes() []byte {
return sdk.MustSortJSON(msgCdc.MustMarshalJSON(msg))
}
// Implements Msg.
func (msg MsgSend) GetSigners() []sdk.AccAddress {
return []sdk.AccAddress{msg.FromAddress}
}
// MsgMultiSend - high level transaction of the coin module
type MsgMultiSend struct {
Inputs []Input `json:"inputs"`
Outputs []Output `json:"outputs"`
}
var _ sdk.Msg = MsgMultiSend{}
// NewMsgMultiSend - construct arbitrary multi-in, multi-out send msg.
func NewMsgMultiSend(in []Input, out []Output) MsgMultiSend {
return MsgMultiSend{Inputs: in, Outputs: out}
}
// Implements Msg.
// nolint
func (msg MsgMultiSend) Route() string { return RouterKey }
func (msg MsgMultiSend) Type() string { return "multisend" }
// Implements Msg.
func (msg MsgMultiSend) ValidateBasic() sdk.Error {
// this just makes sure all the inputs and outputs are properly formatted, // this just makes sure all the inputs and outputs are properly formatted,
// not that they actually have the money inside // not that they actually have the money inside
if len(msg.Inputs) == 0 { if len(msg.Inputs) == 0 {
@ -40,13 +83,12 @@ func (msg MsgSend) ValidateBasic() sdk.Error {
} }
// Implements Msg. // Implements Msg.
func (msg MsgSend) GetSignBytes() []byte { func (msg MsgMultiSend) GetSignBytes() []byte {
bz := msgCdc.MustMarshalJSON(msg) return sdk.MustSortJSON(msgCdc.MustMarshalJSON(msg))
return sdk.MustSortJSON(bz)
} }
// Implements Msg. // Implements Msg.
func (msg MsgSend) GetSigners() []sdk.AccAddress { func (msg MsgMultiSend) GetSigners() []sdk.AccAddress {
addrs := make([]sdk.AccAddress, len(msg.Inputs)) addrs := make([]sdk.AccAddress, len(msg.Inputs))
for i, in := range msg.Inputs { for i, in := range msg.Inputs {
addrs[i] = in.Address addrs[i] = in.Address
@ -77,7 +119,7 @@ func (in Input) ValidateBasic() sdk.Error {
return nil return nil
} }
// NewInput - create a transaction input, used with MsgSend // NewInput - create a transaction input, used with MsgMultiSend
func NewInput(addr sdk.AccAddress, coins sdk.Coins) Input { func NewInput(addr sdk.AccAddress, coins sdk.Coins) Input {
return Input{ return Input{
Address: addr, Address: addr,
@ -108,7 +150,7 @@ func (out Output) ValidateBasic() sdk.Error {
return nil return nil
} }
// NewOutput - create a transaction output, used with MsgSend // NewOutput - create a transaction output, used with MsgMultiSend
func NewOutput(addr sdk.AccAddress, coins sdk.Coins) Output { func NewOutput(addr sdk.AccAddress, coins sdk.Coins) Output {
return Output{ return Output{
Address: addr, Address: addr,

View File

@ -9,20 +9,79 @@ import (
sdk "github.com/cosmos/cosmos-sdk/types" sdk "github.com/cosmos/cosmos-sdk/types"
) )
func TestNewMsgSend(t *testing.T) {}
func TestMsgSendRoute(t *testing.T) { func TestMsgSendRoute(t *testing.T) {
addr1 := sdk.AccAddress([]byte("from"))
addr2 := sdk.AccAddress([]byte("to"))
coins := sdk.Coins{sdk.NewInt64Coin("atom", 10)}
var msg = NewMsgSend(addr1, addr2, coins)
require.Equal(t, msg.Route(), "bank")
require.Equal(t, msg.Type(), "send")
}
func TestMsgSendValidation(t *testing.T) {
addr1 := sdk.AccAddress([]byte("from"))
addr2 := sdk.AccAddress([]byte("to"))
atom123 := sdk.Coins{sdk.NewInt64Coin("atom", 123)}
atom0 := sdk.Coins{sdk.NewInt64Coin("atom", 0)}
atom123eth123 := sdk.Coins{sdk.NewInt64Coin("atom", 123), sdk.NewInt64Coin("eth", 123)}
atom123eth0 := sdk.Coins{sdk.NewInt64Coin("atom", 123), sdk.NewInt64Coin("eth", 0)}
var emptyAddr sdk.AccAddress
cases := []struct {
valid bool
tx MsgSend
}{
{true, NewMsgSend(addr1, addr2, atom123)}, // valid send
{true, NewMsgSend(addr1, addr2, atom123eth123)}, // valid send with multiple coins
{false, NewMsgSend(addr1, addr2, atom0)}, // non positive coin
{false, NewMsgSend(addr1, addr2, atom123eth0)}, // non positive coin in multicoins
{false, NewMsgSend(emptyAddr, addr2, atom123)}, // empty from addr
{false, NewMsgSend(addr1, emptyAddr, atom123)}, // empty to addr
}
for i, tc := range cases {
err := tc.tx.ValidateBasic()
if tc.valid {
require.Nil(t, err, "%d: %+v", i, err)
} else {
require.NotNil(t, err, "%d", i)
}
}
}
func TestMsgSendGetSignBytes(t *testing.T) {
addr1 := sdk.AccAddress([]byte("input"))
addr2 := sdk.AccAddress([]byte("output"))
coins := sdk.Coins{sdk.NewInt64Coin("atom", 10)}
var msg = NewMsgSend(addr1, addr2, coins)
res := msg.GetSignBytes()
expected := `{"type":"cosmos-sdk/Send","value":{"amount":[{"amount":"10","denom":"atom"}],"from_address":"cosmos1d9h8qat57ljhcm","to_address":"cosmos1da6hgur4wsmpnjyg"}}`
require.Equal(t, expected, string(res))
}
func TestMsgSendGetSigners(t *testing.T) {
var msg = NewMsgSend(sdk.AccAddress([]byte("input1")), sdk.AccAddress{}, sdk.Coins{})
res := msg.GetSigners()
// TODO: fix this !
require.Equal(t, fmt.Sprintf("%v", res), "[696E70757431]")
}
func TestMsgMultiSendRoute(t *testing.T) {
// Construct a MsgSend // Construct a MsgSend
addr1 := sdk.AccAddress([]byte("input")) addr1 := sdk.AccAddress([]byte("input"))
addr2 := sdk.AccAddress([]byte("output")) addr2 := sdk.AccAddress([]byte("output"))
coins := sdk.Coins{sdk.NewInt64Coin("atom", 10)} coins := sdk.Coins{sdk.NewInt64Coin("atom", 10)}
var msg = MsgSend{ var msg = MsgMultiSend{
Inputs: []Input{NewInput(addr1, coins)}, Inputs: []Input{NewInput(addr1, coins)},
Outputs: []Output{NewOutput(addr2, coins)}, Outputs: []Output{NewOutput(addr2, coins)},
} }
// TODO some failures for bad result // TODO some failures for bad result
require.Equal(t, msg.Route(), "bank") require.Equal(t, msg.Route(), "bank")
require.Equal(t, msg.Type(), "multisend")
} }
func TestInputValidation(t *testing.T) { func TestInputValidation(t *testing.T) {
@ -101,7 +160,7 @@ func TestOutputValidation(t *testing.T) {
} }
} }
func TestMsgSendValidation(t *testing.T) { func TestMsgMultiSendValidation(t *testing.T) {
addr1 := sdk.AccAddress([]byte{1, 2}) addr1 := sdk.AccAddress([]byte{1, 2})
addr2 := sdk.AccAddress([]byte{7, 8}) addr2 := sdk.AccAddress([]byte{7, 8})
atom123 := sdk.Coins{sdk.NewInt64Coin("atom", 123)} atom123 := sdk.Coins{sdk.NewInt64Coin("atom", 123)}
@ -120,40 +179,40 @@ func TestMsgSendValidation(t *testing.T) {
cases := []struct { cases := []struct {
valid bool valid bool
tx MsgSend tx MsgMultiSend
}{ }{
{false, MsgSend{}}, // no input or output {false, MsgMultiSend{}}, // no input or output
{false, MsgSend{Inputs: []Input{input1}}}, // just input {false, MsgMultiSend{Inputs: []Input{input1}}}, // just input
{false, MsgSend{Outputs: []Output{output1}}}, // just output {false, MsgMultiSend{Outputs: []Output{output1}}}, // just output
{false, MsgSend{ {false, MsgMultiSend{
Inputs: []Input{NewInput(emptyAddr, atom123)}, // invalid input Inputs: []Input{NewInput(emptyAddr, atom123)}, // invalid input
Outputs: []Output{output1}}}, Outputs: []Output{output1}}},
{false, MsgSend{ {false, MsgMultiSend{
Inputs: []Input{input1}, Inputs: []Input{input1},
Outputs: []Output{{emptyAddr, atom123}}}, // invalid output Outputs: []Output{{emptyAddr, atom123}}}, // invalid output
}, },
{false, MsgSend{ {false, MsgMultiSend{
Inputs: []Input{input1}, Inputs: []Input{input1},
Outputs: []Output{output2}}, // amounts dont match Outputs: []Output{output2}}, // amounts dont match
}, },
{false, MsgSend{ {false, MsgMultiSend{
Inputs: []Input{input1}, Inputs: []Input{input1},
Outputs: []Output{output3}}, // amounts dont match Outputs: []Output{output3}}, // amounts dont match
}, },
{false, MsgSend{ {false, MsgMultiSend{
Inputs: []Input{input1}, Inputs: []Input{input1},
Outputs: []Output{outputMulti}}, // amounts dont match Outputs: []Output{outputMulti}}, // amounts dont match
}, },
{false, MsgSend{ {false, MsgMultiSend{
Inputs: []Input{input2}, Inputs: []Input{input2},
Outputs: []Output{output1}}, // amounts dont match Outputs: []Output{output1}}, // amounts dont match
}, },
{true, MsgSend{ {true, MsgMultiSend{
Inputs: []Input{input1}, Inputs: []Input{input1},
Outputs: []Output{output1}}, Outputs: []Output{output1}},
}, },
{true, MsgSend{ {true, MsgMultiSend{
Inputs: []Input{input1, input2}, Inputs: []Input{input1, input2},
Outputs: []Output{outputMulti}}, Outputs: []Output{outputMulti}},
}, },
@ -169,22 +228,22 @@ func TestMsgSendValidation(t *testing.T) {
} }
} }
func TestMsgSendGetSignBytes(t *testing.T) { func TestMsgMultiSendGetSignBytes(t *testing.T) {
addr1 := sdk.AccAddress([]byte("input")) addr1 := sdk.AccAddress([]byte("input"))
addr2 := sdk.AccAddress([]byte("output")) addr2 := sdk.AccAddress([]byte("output"))
coins := sdk.Coins{sdk.NewInt64Coin("atom", 10)} coins := sdk.Coins{sdk.NewInt64Coin("atom", 10)}
var msg = MsgSend{ var msg = MsgMultiSend{
Inputs: []Input{NewInput(addr1, coins)}, Inputs: []Input{NewInput(addr1, coins)},
Outputs: []Output{NewOutput(addr2, coins)}, Outputs: []Output{NewOutput(addr2, coins)},
} }
res := msg.GetSignBytes() res := msg.GetSignBytes()
expected := `{"type":"cosmos-sdk/Send","value":{"inputs":[{"address":"cosmos1d9h8qat57ljhcm","coins":[{"amount":"10","denom":"atom"}]}],"outputs":[{"address":"cosmos1da6hgur4wsmpnjyg","coins":[{"amount":"10","denom":"atom"}]}]}}` expected := `{"type":"cosmos-sdk/MultiSend","value":{"inputs":[{"address":"cosmos1d9h8qat57ljhcm","coins":[{"amount":"10","denom":"atom"}]}],"outputs":[{"address":"cosmos1da6hgur4wsmpnjyg","coins":[{"amount":"10","denom":"atom"}]}]}}`
require.Equal(t, expected, string(res)) require.Equal(t, expected, string(res))
} }
func TestMsgSendGetSigners(t *testing.T) { func TestMsgMultiSendGetSigners(t *testing.T) {
var msg = MsgSend{ var msg = MsgMultiSend{
Inputs: []Input{ Inputs: []Input{
NewInput(sdk.AccAddress([]byte("input1")), nil), NewInput(sdk.AccAddress([]byte("input1")), nil),
NewInput(sdk.AccAddress([]byte("input2")), nil), NewInput(sdk.AccAddress([]byte("input2")), nil),

View File

@ -16,11 +16,30 @@ import (
"github.com/cosmos/cosmos-sdk/x/mock/simulation" "github.com/cosmos/cosmos-sdk/x/mock/simulation"
) )
// SingleInputSendTx tests and runs a single msg send w/ auth, with one input and one output, where both // SendTx tests and runs a single msg send where both
// accounts already exist. // accounts already exist.
func SingleInputSendTx(mapper auth.AccountKeeper) simulation.Operation { func SendMsg(mapper auth.AccountKeeper, bk bank.Keeper) simulation.Operation {
handler := bank.NewHandler(bk)
return func(r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context, accs []simulation.Account, event func(string)) (action string, fOps []simulation.FutureOperation, err error) { return func(r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context, accs []simulation.Account, event func(string)) (action string, fOps []simulation.FutureOperation, err error) {
fromAcc, action, msg, abort := createSingleInputSendMsg(r, ctx, accs, mapper) fromAcc, action, msg, abort := createSendMsg(r, ctx, accs, mapper)
if abort {
return action, nil, nil
}
err = sendAndVerifyMsgSend(app, mapper, msg, ctx, []crypto.PrivKey{fromAcc.PrivKey}, handler)
if err != nil {
return "", nil, err
}
event("bank/sendAndVerifyTxSend/ok")
return action, nil, nil
}
}
// SendTx tests and runs a single tx send, with auth where both
// accounts already exist.
func SendTx(mapper auth.AccountKeeper) simulation.Operation {
return func(r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context, accs []simulation.Account, event func(string)) (action string, fOps []simulation.FutureOperation, err error) {
fromAcc, action, msg, abort := createSendMsg(r, ctx, accs, mapper)
if abort { if abort {
return action, nil, nil return action, nil, nil
} }
@ -34,26 +53,7 @@ func SingleInputSendTx(mapper auth.AccountKeeper) simulation.Operation {
} }
} }
// SingleInputSendMsg tests and runs a single msg send, with one input and one output, where both func createSendMsg(r *rand.Rand, ctx sdk.Context, accs []simulation.Account, mapper auth.AccountKeeper) (fromAcc simulation.Account, action string, msg bank.MsgSend, abort bool) {
// accounts already exist.
func SingleInputSendMsg(mapper auth.AccountKeeper, bk bank.Keeper) simulation.Operation {
handler := bank.NewHandler(bk)
return func(r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context, accs []simulation.Account, event func(string)) (action string, fOps []simulation.FutureOperation, err error) {
fromAcc, action, msg, abort := createSingleInputSendMsg(r, ctx, accs, mapper)
if abort {
return action, nil, nil
}
err = sendAndVerifyMsgSend(app, mapper, msg, ctx, []crypto.PrivKey{fromAcc.PrivKey}, handler)
if err != nil {
return "", nil, err
}
event("bank/sendAndVerifyMsgSend/ok")
return action, nil, nil
}
}
func createSingleInputSendMsg(r *rand.Rand, ctx sdk.Context, accs []simulation.Account, mapper auth.AccountKeeper) (fromAcc simulation.Account, action string, msg bank.MsgSend, abort bool) {
fromAcc = simulation.RandomAcc(r, accs) fromAcc = simulation.RandomAcc(r, accs)
toAcc := simulation.RandomAcc(r, accs) toAcc := simulation.RandomAcc(r, accs)
// Disallow sending money to yourself // Disallow sending money to yourself
@ -84,16 +84,133 @@ func createSingleInputSendMsg(r *rand.Rand, ctx sdk.Context, accs []simulation.A
) )
coins := sdk.Coins{sdk.NewCoin(initFromCoins[denomIndex].Denom, amt)} coins := sdk.Coins{sdk.NewCoin(initFromCoins[denomIndex].Denom, amt)}
msg = bank.MsgSend{ msg = bank.NewMsgSend(fromAcc.Address, toAcc.Address, coins)
return
}
// Sends and verifies the transition of a msg send.
func sendAndVerifyMsgSend(app *baseapp.BaseApp, mapper auth.AccountKeeper, msg bank.MsgSend, ctx sdk.Context, privkeys []crypto.PrivKey, handler sdk.Handler) error {
fromAcc := mapper.GetAccount(ctx, msg.FromAddress)
AccountNumbers := []uint64{fromAcc.GetAccountNumber()}
SequenceNumbers := []uint64{fromAcc.GetSequence()}
initialFromAddrCoins := fromAcc.GetCoins()
toAcc := mapper.GetAccount(ctx, msg.ToAddress)
initialToAddrCoins := toAcc.GetCoins()
if handler != nil {
res := handler(ctx, msg)
if !res.IsOK() {
if res.Code == bank.CodeSendDisabled {
return nil
}
// TODO: Do this in a more 'canonical' way
return fmt.Errorf("handling msg failed %v", res)
}
} else {
tx := mock.GenTx([]sdk.Msg{msg},
AccountNumbers,
SequenceNumbers,
privkeys...)
res := app.Deliver(tx)
if !res.IsOK() {
// TODO: Do this in a more 'canonical' way
return fmt.Errorf("Deliver failed %v", res)
}
}
fromAcc = mapper.GetAccount(ctx, msg.FromAddress)
toAcc = mapper.GetAccount(ctx, msg.ToAddress)
if !initialFromAddrCoins.Minus(msg.Amount).IsEqual(fromAcc.GetCoins()) {
return fmt.Errorf("fromAddress %s had an incorrect amount of coins", fromAcc.GetAddress())
}
if !initialToAddrCoins.Plus(msg.Amount).IsEqual(toAcc.GetCoins()) {
return fmt.Errorf("toAddress %s had an incorrect amount of coins", toAcc.GetAddress())
}
return nil
}
// SingleInputSendTx tests and runs a single msg multisend w/ auth, with one input and one output, where both
// accounts already exist.
func SingleInputMultiSendTx(mapper auth.AccountKeeper) simulation.Operation {
return func(r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context, accs []simulation.Account, event func(string)) (action string, fOps []simulation.FutureOperation, err error) {
fromAcc, action, msg, abort := createSingleInputMsgMultiSend(r, ctx, accs, mapper)
if abort {
return action, nil, nil
}
err = sendAndVerifyMsgMultiSend(app, mapper, msg, ctx, []crypto.PrivKey{fromAcc.PrivKey}, nil)
if err != nil {
return "", nil, err
}
event("bank/sendAndVerifyTxMultiSend/ok")
return action, nil, nil
}
}
// SingleInputSendMsg tests and runs a single msg multisend, with one input and one output, where both
// accounts already exist.
func SingleInputMsgMultiSend(mapper auth.AccountKeeper, bk bank.Keeper) simulation.Operation {
handler := bank.NewHandler(bk)
return func(r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context, accs []simulation.Account, event func(string)) (action string, fOps []simulation.FutureOperation, err error) {
fromAcc, action, msg, abort := createSingleInputMsgMultiSend(r, ctx, accs, mapper)
if abort {
return action, nil, nil
}
err = sendAndVerifyMsgMultiSend(app, mapper, msg, ctx, []crypto.PrivKey{fromAcc.PrivKey}, handler)
if err != nil {
return "", nil, err
}
event("bank/sendAndVerifyMsgMultiSend/ok")
return action, nil, nil
}
}
func createSingleInputMsgMultiSend(r *rand.Rand, ctx sdk.Context, accs []simulation.Account, mapper auth.AccountKeeper) (fromAcc simulation.Account, action string, msg bank.MsgMultiSend, abort bool) {
fromAcc = simulation.RandomAcc(r, accs)
toAcc := simulation.RandomAcc(r, accs)
// Disallow sending money to yourself
for {
if !fromAcc.PubKey.Equals(toAcc.PubKey) {
break
}
toAcc = simulation.RandomAcc(r, accs)
}
toAddr := toAcc.Address
initFromCoins := mapper.GetAccount(ctx, fromAcc.Address).SpendableCoins(ctx.BlockHeader().Time)
if len(initFromCoins) == 0 {
return fromAcc, "skipping, no coins at all", msg, true
}
denomIndex := r.Intn(len(initFromCoins))
amt, goErr := randPositiveInt(r, initFromCoins[denomIndex].Amount)
if goErr != nil {
return fromAcc, "skipping bank send due to account having no coins of denomination " + initFromCoins[denomIndex].Denom, msg, true
}
action = fmt.Sprintf("%s is sending %s %s to %s",
fromAcc.Address.String(),
amt.String(),
initFromCoins[denomIndex].Denom,
toAddr.String(),
)
coins := sdk.Coins{sdk.NewCoin(initFromCoins[denomIndex].Denom, amt)}
msg = bank.MsgMultiSend{
Inputs: []bank.Input{bank.NewInput(fromAcc.Address, coins)}, Inputs: []bank.Input{bank.NewInput(fromAcc.Address, coins)},
Outputs: []bank.Output{bank.NewOutput(toAddr, coins)}, Outputs: []bank.Output{bank.NewOutput(toAddr, coins)},
} }
return return
} }
// Sends and verifies the transition of a msg send. This fails if there are repeated inputs or outputs // Sends and verifies the transition of a msg multisend. This fails if there are repeated inputs or outputs
// pass in handler as nil to handle txs, otherwise handle msgs // pass in handler as nil to handle txs, otherwise handle msgs
func sendAndVerifyMsgSend(app *baseapp.BaseApp, mapper auth.AccountKeeper, msg bank.MsgSend, ctx sdk.Context, privkeys []crypto.PrivKey, handler sdk.Handler) error { func sendAndVerifyMsgMultiSend(app *baseapp.BaseApp, mapper auth.AccountKeeper, msg bank.MsgMultiSend, ctx sdk.Context, privkeys []crypto.PrivKey, handler sdk.Handler) error {
initialInputAddrCoins := make([]sdk.Coins, len(msg.Inputs)) initialInputAddrCoins := make([]sdk.Coins, len(msg.Inputs))
initialOutputAddrCoins := make([]sdk.Coins, len(msg.Outputs)) initialOutputAddrCoins := make([]sdk.Coins, len(msg.Outputs))
AccountNumbers := make([]uint64, len(msg.Inputs)) AccountNumbers := make([]uint64, len(msg.Inputs))