package cli_test import ( "context" "fmt" "io" "testing" "github.com/stretchr/testify/suite" sdkmath "cosmossdk.io/math" "cosmossdk.io/x/bank" "cosmossdk.io/x/bank/client/cli" "github.com/cosmos/cosmos-sdk/client" "github.com/cosmos/cosmos-sdk/client/flags" addresscodec "github.com/cosmos/cosmos-sdk/codec/address" codectestutil "github.com/cosmos/cosmos-sdk/codec/testutil" "github.com/cosmos/cosmos-sdk/crypto/keyring" "github.com/cosmos/cosmos-sdk/testutil" clitestutil "github.com/cosmos/cosmos-sdk/testutil/cli" sdk "github.com/cosmos/cosmos-sdk/types" testutilmod "github.com/cosmos/cosmos-sdk/types/module/testutil" ) type CLITestSuite struct { suite.Suite kr keyring.Keyring encCfg testutilmod.TestEncodingConfig baseCtx client.Context } func TestCLITestSuite(t *testing.T) { suite.Run(t, new(CLITestSuite)) } func (s *CLITestSuite) SetupSuite() { s.encCfg = testutilmod.MakeTestEncodingConfig(codectestutil.CodecOptions{}, bank.AppModule{}) s.kr = keyring.NewInMemory(s.encCfg.Codec) s.baseCtx = client.Context{}. WithKeyring(s.kr). WithTxConfig(s.encCfg.TxConfig). WithCodec(s.encCfg.Codec). WithClient(clitestutil.MockCometRPC{}). WithAccountRetriever(client.MockAccountRetriever{}). WithOutput(io.Discard). WithAddressCodec(addresscodec.NewBech32Codec("cosmos")). WithValidatorAddressCodec(addresscodec.NewBech32Codec("cosmosvaloper")). WithConsensusAddressCodec(addresscodec.NewBech32Codec("cosmosvalcons")) } func (s *CLITestSuite) TestMultiSendTxCmd() { accounts := testutil.CreateKeyringAccounts(s.T(), s.kr, 3) accountStr := make([]string, len(accounts)) for i, acc := range accounts { addrStr, err := s.baseCtx.AddressCodec.BytesToString(acc.Address) s.Require().NoError(err) accountStr[i] = addrStr } cmd := cli.NewMultiSendTxCmd() cmd.SetOutput(io.Discard) extraArgs := []string{ fmt.Sprintf("--%s=%s", flags.FlagBroadcastMode, flags.BroadcastSync), fmt.Sprintf("--%s=true", flags.FlagSkipConfirmation), fmt.Sprintf("--%s=%s", flags.FlagBroadcastMode, flags.BroadcastSync), fmt.Sprintf("--%s=%s", flags.FlagFees, sdk.NewCoins(sdk.NewCoin("photon", sdkmath.NewInt(10))).String()), fmt.Sprintf("--%s=test-chain", flags.FlagChainID), } testCases := []struct { name string ctxGen func() client.Context from string to []string amount sdk.Coins extraArgs []string expectErrMsg string }{ { "valid transaction", func() client.Context { return s.baseCtx }, accountStr[0], []string{ accountStr[1], accountStr[2], }, sdk.NewCoins( sdk.NewCoin("stake", sdkmath.NewInt(10)), sdk.NewCoin("photon", sdkmath.NewInt(40)), ), extraArgs, "", }, { "invalid from Address", func() client.Context { return s.baseCtx }, "foo", []string{ accountStr[1], accountStr[2], }, sdk.NewCoins( sdk.NewCoin("stake", sdkmath.NewInt(10)), sdk.NewCoin("photon", sdkmath.NewInt(40)), ), extraArgs, "key not found", }, { "invalid recipients", func() client.Context { return s.baseCtx }, accountStr[0], []string{ accountStr[1], "bar", }, sdk.NewCoins( sdk.NewCoin("stake", sdkmath.NewInt(10)), sdk.NewCoin("photon", sdkmath.NewInt(40)), ), extraArgs, "invalid bech32 string", }, { "invalid amount", func() client.Context { return s.baseCtx }, accountStr[0], []string{ accountStr[1], accountStr[2], }, nil, extraArgs, "must send positive amount", }, } for _, tc := range testCases { s.Run(tc.name, func() { var args []string args = append(args, tc.from) args = append(args, tc.to...) args = append(args, tc.amount.String()) args = append(args, tc.extraArgs...) cmd.SetContext(context.WithValue(context.Background(), client.ClientContextKey, &client.Context{})) cmd.SetArgs(args) s.Require().NoError(client.SetCmdClientContextHandler(tc.ctxGen(), cmd)) out, err := clitestutil.ExecTestCLICmd(tc.ctxGen(), cmd, args) if tc.expectErrMsg != "" { s.Require().Error(err) s.Require().Contains(out.String(), tc.expectErrMsg) } else { s.Require().NoError(err) msg := &sdk.TxResponse{} s.Require().NoError(tc.ctxGen().Codec.UnmarshalJSON(out.Bytes(), msg), out.String()) } }) } }