Merge PR #3428: SendMsg and MultiSendMsg seperation
This commit is contained in:
parent
082295172e
commit
18eee0a3a8
|
@ -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) {
|
||||||
|
|
|
@ -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)},
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)},
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
|
||||||
}
|
|
|
@ -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()
|
||||||
|
|
|
@ -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()
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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),
|
||||||
|
|
|
@ -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))
|
||||||
|
|
Loading…
Reference in New Issue