package stake import ( "testing" sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/x/auth" "github.com/cosmos/cosmos-sdk/x/auth/mock" "github.com/cosmos/cosmos-sdk/x/bank" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" abci "github.com/tendermint/tendermint/abci/types" "github.com/tendermint/tendermint/crypto" ) var ( priv1 = crypto.GenPrivKeyEd25519() addr1 = priv1.PubKey().Address() priv2 = crypto.GenPrivKeyEd25519() addr2 = priv2.PubKey().Address() addr3 = crypto.GenPrivKeyEd25519().PubKey().Address() priv4 = crypto.GenPrivKeyEd25519() addr4 = priv4.PubKey().Address() coins = sdk.Coins{{"foocoin", sdk.NewInt(10)}} fee = auth.StdFee{ sdk.Coins{{"foocoin", sdk.NewInt(0)}}, 100000, } ) // initialize the mock application for this module func getMockApp(t *testing.T) (*mock.App, Keeper) { mapp := mock.NewApp() RegisterWire(mapp.Cdc) keyStake := sdk.NewKVStoreKey("stake") coinKeeper := bank.NewKeeper(mapp.AccountMapper) keeper := NewKeeper(mapp.Cdc, keyStake, coinKeeper, mapp.RegisterCodespace(DefaultCodespace)) mapp.Router().AddRoute("stake", NewHandler(keeper)) mapp.SetEndBlocker(getEndBlocker(keeper)) mapp.SetInitChainer(getInitChainer(mapp, keeper)) require.NoError(t, mapp.CompleteSetup([]*sdk.KVStoreKey{keyStake})) return mapp, keeper } // stake endblocker func getEndBlocker(keeper Keeper) sdk.EndBlocker { return func(ctx sdk.Context, req abci.RequestEndBlock) abci.ResponseEndBlock { validatorUpdates := EndBlocker(ctx, keeper) return abci.ResponseEndBlock{ ValidatorUpdates: validatorUpdates, } } } // overwrite the mock init chainer func getInitChainer(mapp *mock.App, keeper Keeper) sdk.InitChainer { return func(ctx sdk.Context, req abci.RequestInitChain) abci.ResponseInitChain { mapp.InitChainer(ctx, req) stakeGenesis := DefaultGenesisState() stakeGenesis.Pool.LooseTokens = 100000 InitGenesis(ctx, keeper, stakeGenesis) return abci.ResponseInitChain{} } } //__________________________________________________________________________________________ func checkValidator(t *testing.T, mapp *mock.App, keeper Keeper, addr sdk.Address, expFound bool) Validator { ctxCheck := mapp.BaseApp.NewContext(true, abci.Header{}) validator, found := keeper.GetValidator(ctxCheck, addr1) require.Equal(t, expFound, found) return validator } func checkDelegation(t *testing.T, mapp *mock.App, keeper Keeper, delegatorAddr, validatorAddr sdk.Address, expFound bool, expShares sdk.Rat) { ctxCheck := mapp.BaseApp.NewContext(true, abci.Header{}) delegation, found := keeper.GetDelegation(ctxCheck, delegatorAddr, validatorAddr) if expFound { require.True(t, found) assert.True(sdk.RatEq(t, expShares, delegation.Shares)) return } require.False(t, found) } func TestStakeMsgs(t *testing.T) { mapp, keeper := getMockApp(t) genCoin := sdk.Coin{"steak", sdk.NewInt(42)} bondCoin := sdk.Coin{"steak", sdk.NewInt(10)} acc1 := &auth.BaseAccount{ Address: addr1, Coins: sdk.Coins{genCoin}, } acc2 := &auth.BaseAccount{ Address: addr2, Coins: sdk.Coins{genCoin}, } accs := []auth.Account{acc1, acc2} mock.SetGenesis(mapp, accs) mock.CheckBalance(t, mapp, addr1, sdk.Coins{genCoin}) mock.CheckBalance(t, mapp, addr2, sdk.Coins{genCoin}) //////////////////// // Create Validator description := NewDescription("foo_moniker", "", "", "") createValidatorMsg := NewMsgCreateValidator( addr1, priv1.PubKey(), bondCoin, description, ) mock.SignCheckDeliver(t, mapp.BaseApp, []sdk.Msg{createValidatorMsg}, []int64{0}, []int64{0}, true, priv1) mock.CheckBalance(t, mapp, addr1, sdk.Coins{genCoin.Minus(bondCoin)}) mapp.BeginBlock(abci.RequestBeginBlock{}) validator := checkValidator(t, mapp, keeper, addr1, true) require.Equal(t, addr1, validator.Owner) require.Equal(t, sdk.Bonded, validator.Status()) require.True(sdk.RatEq(t, sdk.NewRat(10), validator.PoolShares.Bonded())) // check the bond that should have been created as well checkDelegation(t, mapp, keeper, addr1, addr1, true, sdk.NewRat(10)) //////////////////// // Edit Validator description = NewDescription("bar_moniker", "", "", "") editValidatorMsg := NewMsgEditValidator(addr1, description) mock.SignCheckDeliver(t, mapp.BaseApp, []sdk.Msg{editValidatorMsg}, []int64{0}, []int64{1}, true, priv1) validator = checkValidator(t, mapp, keeper, addr1, true) require.Equal(t, description, validator.Description) //////////////////// // Delegate mock.CheckBalance(t, mapp, addr2, sdk.Coins{genCoin}) delegateMsg := NewMsgDelegate(addr2, addr1, bondCoin) mock.SignCheckDeliver(t, mapp.BaseApp, []sdk.Msg{delegateMsg}, []int64{1}, []int64{0}, true, priv2) mock.CheckBalance(t, mapp, addr2, sdk.Coins{genCoin.Minus(bondCoin)}) checkDelegation(t, mapp, keeper, addr2, addr1, true, sdk.NewRat(10)) //////////////////// // Begin Unbonding beginUnbondingMsg := NewMsgBeginUnbonding(addr2, addr1, sdk.NewRat(10)) mock.SignCheckDeliver(t, mapp.BaseApp, []sdk.Msg{beginUnbondingMsg}, []int64{1}, []int64{1}, true, priv2) // delegation should exist anymore checkDelegation(t, mapp, keeper, addr2, addr1, false, sdk.Rat{}) // balance should be the same because bonding not yet complete mock.CheckBalance(t, mapp, addr2, sdk.Coins{genCoin.Minus(bondCoin)}) }