mirror of
https://github.com/0glabs/0g-chain.git
synced 2024-12-26 00:05:18 +00:00
f757d7ab15
* Update cosmos-sdk to v0.45.10-kava * Add RegisterNodeService to app * Update cosmos proto files * Update cosmos proto files * Use tagged v0.45.10-kava-v0.19-0.21 cosmos version * update x/auth/legacy to x/auth/migrations * Delete rest packages and registration * Remove rest from proposal handlers * Remove legacy types referencing removed sdk types * Remove legacy tx broadcast handler * Update incentive staking hooks to return error * Remove grpc replace directive, use new grpc version * Fix storetypes import * Update tally_handler with updated gov types * Delete legacy types * Use new gov default config * Update RegisterTendermintService params Signed-off-by: drklee3 <derrick@dlee.dev> * Replace sdk.StoreKey with storetypes.StoreKey * Replace sdk.Int#ToDec with sdk.NewDecFromInt * Replace sdk.NewUintFromBigInt with sdkmath.NewUintFromBigInt Signed-off-by: drklee3 <derrick@dlee.dev> * Update most intances of govtypes to govv1beta1 * Unpack coin slice for Coins#Sub and Coins#SafeSub Signed-off-by: drklee3 <derrick@dlee.dev> * Update committee gov codec registration Signed-off-by: drklee3 <derrick@dlee.dev> * Update migrate utils period_vesting Coins#Sub Signed-off-by: drklee3 <derrick@dlee.dev> * Update Coin#Sub in community proposal handler Signed-off-by: drklee3 <derrick@dlee.dev> * Update Coin#Sub, FundModuleAccount/FundAccount in banktestutil Signed-off-by: drklee3 <derrick@dlee.dev> * Update community, earn, kavadist proposal gov registration * Update evm cli client EthSecp256k1Type check * AccAddressFromHex to AccAddressFromHexUnsafe * Add mint DefaultInflationCalculationFn to earn test * Update use of removed staking.NewHandler * Rename FlagIAVLFastNode -> FlagDisableIAVLFastNode * cmd: Update new snapshot app option Signed-off-by: drklee3 <derrick@dlee.dev> * cmd: Add tendermint default config, use cosmos rpc status command Signed-off-by: drklee3 <derrick@dlee.dev> * Update ethermint import path github.com/tharsis/ethermint -> github.com/evmos/ethermint * Upgrade ibc-go to v6 * Update proto dependencies Signed-off-by: drklee3 <derrick@dlee.dev> * Update Tally handler test with new gov types * Update helpers.GenTx -> helpers.GenSignedMockTx * Update evmkeeper.NewKeeper params Signed-off-by: drklee3 <derrick@dlee.dev> * Update ante authz, tests * Add feemarket transient key, pass subspaces to evm/feemarket keepers * Update new ante decorators * Add new addModuleInitFlags to server commands * Pass codec to keyring.New in genaccounts * Pass codec to client keys add * Add SendCoins to evmutil bank_keeper * Use github.com/cosmos/iavl@v0.19.5 * Add ante HandlerOptions * Add unimplemented SendCoins to evmutil bank keeper Ethermint x/evm does not use this method * Update init-new-chain script to disable post-london blocks * Modify test genesis states to append 1 validator * Update tally handler test to use string values * Prevent querying balance for empty sdk.AccAddress in auction bidding test * Set default bond denom to ukava * Remove overwritten bank genesis total supply in committee proposal test Signed-off-by: drklee3 <derrick@dlee.dev> * Use ukava for testing staked balance * Disable minting in community proposal handler test Previously stake denom is used, which resulted in 0 minted coins * Update hard APYToSPY test expected value Increased iterations in sdk.ApproxRoot, updated closer to real value * Fix NewDecCoinsFromCoins bug in incentive collectDerivativeStakingRewards * Allow bkava earn incentive test values to match within small margin for rounding Signed-off-by: drklee3 <derrick@dlee.dev> * Update invalid denom in issuance message coin validation Colons are now valid in denoms Signed-off-by: drklee3 <derrick@dlee.dev> * Remove genesis validator in incentive delegation tests * Update pricefeed market test for invalid denom Signed-off-by: drklee3 <derrick@dlee.dev> * Update incentive delegator rewards test without genesis validator Signed-off-by: drklee3 <derrick@dlee.dev> * Add validator to export test * Clear bank state in minting tests Signed-off-by: drklee3 <derrick@dlee.dev> * Remove validator for no stake tally test Signed-off-by: drklee3 <derrick@dlee.dev> * Clear incentive state before InitGenesis in incentive genesis export test * Update swagger Signed-off-by: drklee3 <derrick@dlee.dev> * Update ethermint version to match replaced version * Remove legacy swagger * Add NewEthEmitEventDecorator * Remove redundant func for AddModuleInitFlags * Remove unused addBankBalanceForAddress func * Add SetIAVLLazyLoading option to app cmd * Use legacy.RegisterAminoMsg for committee msg concrete registration * Remove unnecessary Amino field * Add evm_util bankkeeper SendCoins comment * Update test method ResetBankState to DeleteGenesisValidatorCoins to be more clear * Validate incentive params.RewardsPerSecond to be non-zero * Validate swap pools to disallow colons in token denoms * Register all legacy amino types on gov modulecdc * Remove redundant Comittee interface registration * Pin goleveldb to v1.0.1-0.20210819022825-2ae1ddf74ef7 Causes failed to load state at height errors * Update ethermint to new pinned version with minGasPrices parse error fix * Update cosmos fork dependcy commit to include reverted account constructor patch * Update Cosmos v0.46.11 and cometbft v0.34.27 * Bump minimum go version to 1.19 * Update tendermint proto * Update internal testnet genesis * Move NewCanTransferDecorator before NewEthGasConsumeDecorator * Add hard borrow store tests (#1514) * add store tests for Borrow type * refactor Deposit tests to match * Fix old bep3 tests (#1515) * Update Ethermint to 1b17445 to fix duplicate proto registration * Add custom status command to use snake_case and stdout * Add SetInflation helper * Reduce ambiguity with evm CanSignEthTx error * Remove init genesis validator claim in test * Add disabled evmante.NewMinGasPriceDecorator with x/feemarket note * chore: use tagged versions for Cosmos and Ethermint forks * update kvtool & increase wait for ibc transfer test --------- Signed-off-by: drklee3 <derrick@dlee.dev> Co-authored-by: Ruaridh <rhuairahrighairidh@users.noreply.github.com> Co-authored-by: Robert Pirtle <astropirtle@gmail.com>
413 lines
14 KiB
Go
413 lines
14 KiB
Go
package app
|
|
|
|
import (
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/cosmos/cosmos-sdk/crypto/keys/ed25519"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
authtypes "github.com/cosmos/cosmos-sdk/x/auth/types"
|
|
govkeeper "github.com/cosmos/cosmos-sdk/x/gov/keeper"
|
|
govv1 "github.com/cosmos/cosmos-sdk/x/gov/types/v1"
|
|
govv1beta1 "github.com/cosmos/cosmos-sdk/x/gov/types/v1beta1"
|
|
"github.com/cosmos/cosmos-sdk/x/staking"
|
|
stakingkeeper "github.com/cosmos/cosmos-sdk/x/staking/keeper"
|
|
stakingtypes "github.com/cosmos/cosmos-sdk/x/staking/types"
|
|
"github.com/stretchr/testify/suite"
|
|
tmproto "github.com/tendermint/tendermint/proto/tendermint/types"
|
|
|
|
earntypes "github.com/kava-labs/kava/x/earn/types"
|
|
liquidtypes "github.com/kava-labs/kava/x/liquid/types"
|
|
)
|
|
|
|
// d is an alias for sdk.MustNewDecFromStr
|
|
var d = sdk.MustNewDecFromStr
|
|
|
|
type tallyHandlerSuite struct {
|
|
suite.Suite
|
|
app TestApp
|
|
ctx sdk.Context
|
|
|
|
staking stakingHelper
|
|
|
|
tallier TallyHandler
|
|
}
|
|
|
|
func TestTallyHandlerSuite(t *testing.T) {
|
|
suite.Run(t, new(tallyHandlerSuite))
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) SetupTest() {
|
|
suite.app = NewTestApp()
|
|
suite.app.InitializeFromGenesisStates()
|
|
genesisTime := time.Date(1998, 1, 1, 0, 0, 0, 0, time.UTC)
|
|
suite.ctx = suite.app.NewContext(false, tmproto.Header{Height: 1, Time: genesisTime})
|
|
|
|
suite.staking = stakingHelper{suite.app.GetStakingKeeper()}
|
|
suite.staking.setBondDenom(suite.ctx, "ukava")
|
|
|
|
suite.tallier = NewTallyHandler(
|
|
suite.app.GetGovKeeper(),
|
|
suite.app.GetStakingKeeper(),
|
|
suite.app.GetSavingsKeeper(),
|
|
suite.app.GetEarnKeeper(),
|
|
suite.app.GetLiquidKeeper(),
|
|
suite.app.GetBankKeeper(),
|
|
)
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) TestVotePower_AllSourcesCounted() {
|
|
user := suite.createAccount(suite.newBondCoin(sdk.NewInt(1e9)))
|
|
|
|
validator := suite.delegateToNewBondedValidator(user.GetAddress(), sdk.NewInt(1e9))
|
|
|
|
derivatives := suite.mintDerivative(user.GetAddress(), validator.GetOperator(), sdk.NewInt(500e6))
|
|
|
|
suite.allowBKavaEarnDeposits()
|
|
suite.earnDeposit(
|
|
user.GetAddress(),
|
|
sdk.NewCoin(derivatives.Denom, sdk.NewInt(250e6)),
|
|
)
|
|
|
|
proposal := suite.createProposal()
|
|
suite.voteOnProposal(user.GetAddress(), proposal.Id, govv1beta1.OptionYes)
|
|
|
|
_, _, results := suite.tallier.Tally(suite.ctx, proposal)
|
|
suite.Equal(sdk.NewInt(500e6+250e6+250e6).String(), results.YesCount)
|
|
suite.Equal(sdk.ZeroInt().String(), results.NoCount)
|
|
suite.Equal(sdk.ZeroInt().String(), results.NoWithVetoCount)
|
|
suite.Equal(sdk.ZeroInt().String(), results.AbstainCount)
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) TestVotePower_UserOverridesValidator() {
|
|
user := suite.createAccount(suite.newBondCoin(sdk.NewInt(1e9)))
|
|
|
|
delegated := sdk.NewInt(1e9)
|
|
validator := suite.delegateToNewBondedValidator(user.GetAddress(), delegated)
|
|
selfDelegated := validator.GetTokens().Sub(delegated)
|
|
|
|
derivatives := suite.mintDerivative(user.GetAddress(), validator.GetOperator(), sdk.NewInt(500e6))
|
|
|
|
suite.allowBKavaEarnDeposits()
|
|
suite.earnDeposit(
|
|
user.GetAddress(),
|
|
sdk.NewCoin(derivatives.Denom, sdk.NewInt(250e6)),
|
|
)
|
|
|
|
proposal := suite.createProposal()
|
|
|
|
// Validator votes, inheriting user's stake and bkava.
|
|
suite.voteOnProposal(validator.GetOperator().Bytes(), proposal.Id, govv1beta1.OptionYes)
|
|
|
|
// use wrapped context to discard the state changes
|
|
readOnlyCtx, _ := suite.ctx.CacheContext()
|
|
_, _, results := suite.tallier.Tally(readOnlyCtx, proposal)
|
|
userPower := sdk.NewInt(500e6 + 250e6 + 250e6)
|
|
suite.Equal(
|
|
selfDelegated.Add(userPower).String(),
|
|
results.YesCount,
|
|
)
|
|
suite.Equal(sdk.ZeroInt().String(), results.NoCount)
|
|
suite.Equal(sdk.ZeroInt().String(), results.NoWithVetoCount)
|
|
suite.Equal(sdk.ZeroInt().String(), results.AbstainCount)
|
|
|
|
// User votes, taking power away from validator.
|
|
suite.voteOnProposal(user.GetAddress(), proposal.Id, govv1beta1.OptionNo)
|
|
|
|
_, _, results = suite.tallier.Tally(suite.ctx, proposal)
|
|
suite.Equal(selfDelegated.String(), results.YesCount)
|
|
suite.Equal(userPower.String(), results.NoCount)
|
|
suite.Equal(sdk.ZeroInt().String(), results.NoWithVetoCount)
|
|
suite.Equal(sdk.ZeroInt().String(), results.AbstainCount)
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) TestTallyOutcomes() {
|
|
suite.Run("VotedPowerBelowQuorumFails", func() {
|
|
suite.SetupTest()
|
|
suite.setTallyParams(d("0.4"), d("0.5"), d("0.334"))
|
|
proposal := suite.createProposal()
|
|
|
|
v1 := suite.createNewBondedValidator(sdk.NewInt(399_999_999))
|
|
suite.createNewBondedValidator(sdk.NewInt(600_000_001))
|
|
|
|
suite.voteOnProposal(v1.GetOperator().Bytes(), proposal.Id, govv1beta1.OptionYes)
|
|
|
|
passes, burns, tally := suite.tallier.Tally(suite.ctx, proposal)
|
|
suite.Falsef(passes, "expected proposal to fail, tally: %v", tally)
|
|
suite.Truef(burns, "expected desposit to be burned, tally: %v", tally)
|
|
})
|
|
suite.Run("VetoedFails", func() {
|
|
suite.SetupTest()
|
|
suite.setTallyParams(d("0.4"), d("0.5"), d("0.334"))
|
|
proposal := suite.createProposal()
|
|
|
|
v1 := suite.createNewBondedValidator(sdk.NewInt(334_000_001))
|
|
v2 := suite.createNewBondedValidator(sdk.NewInt(665_999_999))
|
|
|
|
suite.voteOnProposal(v1.GetOperator().Bytes(), proposal.Id, govv1beta1.OptionNoWithVeto)
|
|
suite.voteOnProposal(v2.GetOperator().Bytes(), proposal.Id, govv1beta1.OptionYes)
|
|
|
|
passes, burns, tally := suite.tallier.Tally(suite.ctx, proposal)
|
|
suite.Falsef(passes, "expected proposal to fail, tally: %v", tally)
|
|
suite.Truef(burns, "expected desposit to be burned, tally: %v", tally)
|
|
})
|
|
suite.Run("UnvetoedAndYesAboveThresholdPasses", func() {
|
|
suite.SetupTest()
|
|
suite.setTallyParams(d("0.4"), d("0.5"), d("0.334"))
|
|
proposal := suite.createProposal()
|
|
|
|
v1 := suite.createNewBondedValidator(sdk.NewInt(900_000_000))
|
|
v2 := suite.createNewBondedValidator(sdk.NewInt(50_000_001))
|
|
v3 := suite.createNewBondedValidator(sdk.NewInt(49_999_999))
|
|
|
|
suite.voteOnProposal(v1.GetOperator().Bytes(), proposal.Id, govv1beta1.OptionAbstain)
|
|
suite.voteOnProposal(v2.GetOperator().Bytes(), proposal.Id, govv1beta1.OptionYes)
|
|
suite.voteOnProposal(v3.GetOperator().Bytes(), proposal.Id, govv1beta1.OptionNo)
|
|
|
|
passes, burns, tally := suite.tallier.Tally(suite.ctx, proposal)
|
|
suite.Truef(passes, "expected proposal to pass, tally: %v", tally)
|
|
suite.Falsef(burns, "expected desposit to not burn, tally: %v", tally)
|
|
})
|
|
suite.Run("UnvetoedAndYesBelowThresholdFails", func() {
|
|
suite.SetupTest()
|
|
suite.setTallyParams(d("0.4"), d("0.5"), d("0.334"))
|
|
proposal := suite.createProposal()
|
|
|
|
v1 := suite.createNewBondedValidator(sdk.NewInt(900_000_000))
|
|
v2 := suite.createNewBondedValidator(sdk.NewInt(49_999_999))
|
|
v3 := suite.createNewBondedValidator(sdk.NewInt(50_000_001))
|
|
|
|
suite.voteOnProposal(v1.GetOperator().Bytes(), proposal.Id, govv1beta1.OptionAbstain)
|
|
suite.voteOnProposal(v2.GetOperator().Bytes(), proposal.Id, govv1beta1.OptionYes)
|
|
suite.voteOnProposal(v3.GetOperator().Bytes(), proposal.Id, govv1beta1.OptionNo)
|
|
|
|
passes, burns, tally := suite.tallier.Tally(suite.ctx, proposal)
|
|
suite.Falsef(passes, "expected proposal to pass, tally: %v", tally)
|
|
suite.Falsef(burns, "expected desposit to not burn, tally: %v", tally)
|
|
})
|
|
suite.Run("NotEnoughStakeFails", func() {
|
|
suite.SetupTest()
|
|
suite.setTallyParams(d("0.4"), d("0.5"), d("0.334"))
|
|
proposal := suite.createProposal()
|
|
|
|
// no stake
|
|
suite.app.DeleteGenesisValidator(suite.T(), suite.ctx)
|
|
|
|
passes, burns, tally := suite.tallier.Tally(suite.ctx, proposal)
|
|
suite.Falsef(passes, "expected proposal to pass, tally: %v", tally)
|
|
suite.Falsef(burns, "expected desposit to not burn, tally: %v", tally)
|
|
})
|
|
suite.Run("UnvetoedAndAllAbstainedFails", func() {
|
|
suite.SetupTest()
|
|
suite.setTallyParams(d("0.4"), d("0.5"), d("0.334"))
|
|
proposal := suite.createProposal()
|
|
|
|
v1 := suite.createNewBondedValidator(sdk.NewInt(1e9))
|
|
|
|
suite.voteOnProposal(v1.GetOperator().Bytes(), proposal.Id, govv1beta1.OptionAbstain)
|
|
|
|
passes, burns, tally := suite.tallier.Tally(suite.ctx, proposal)
|
|
suite.Falsef(passes, "expected proposal to pass, tally: %v", tally)
|
|
suite.Falsef(burns, "expected desposit to not burn, tally: %v", tally)
|
|
})
|
|
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) setTallyParams(quorum, threshold, veto sdk.Dec) {
|
|
suite.app.GetGovKeeper().SetTallyParams(suite.ctx, govv1.TallyParams{
|
|
Quorum: quorum.String(),
|
|
Threshold: threshold.String(),
|
|
VetoThreshold: veto.String(),
|
|
})
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) voteOnProposal(
|
|
voter sdk.AccAddress,
|
|
proposalID uint64,
|
|
option govv1beta1.VoteOption,
|
|
) {
|
|
gk := suite.app.GetGovKeeper()
|
|
|
|
err := gk.AddVote(suite.ctx,
|
|
proposalID,
|
|
voter,
|
|
govv1.NewNonSplitVoteOption(govv1.VoteOption(option)),
|
|
"",
|
|
)
|
|
suite.Require().NoError(err)
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) createProposal() govv1.Proposal {
|
|
gk := suite.app.GetGovKeeper()
|
|
deposit := gk.GetDepositParams(suite.ctx).MinDeposit
|
|
proposer := suite.createAccount(deposit...)
|
|
|
|
msg, err := govv1beta1.NewMsgSubmitProposal(
|
|
govv1beta1.NewTextProposal("a title", "a description"),
|
|
deposit,
|
|
proposer.GetAddress(),
|
|
)
|
|
suite.Require().NoError(err)
|
|
|
|
msgServerv1 := govkeeper.NewMsgServerImpl(gk)
|
|
|
|
govAcct := gk.GetGovernanceAccount(suite.ctx).GetAddress()
|
|
msgServer := govkeeper.NewLegacyMsgServerImpl(govAcct.String(), msgServerv1)
|
|
res, err := msgServer.SubmitProposal(sdk.WrapSDKContext(suite.ctx), msg)
|
|
suite.Require().NoError(err)
|
|
|
|
proposal, found := gk.GetProposal(suite.ctx, res.ProposalId)
|
|
if !found {
|
|
panic("proposal not found")
|
|
}
|
|
return proposal
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) newBondCoin(amount sdk.Int) sdk.Coin {
|
|
return suite.staking.newBondCoin(suite.ctx, amount)
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) allowBKavaEarnDeposits() {
|
|
ek := suite.app.GetEarnKeeper()
|
|
earnParams := ek.GetParams(suite.ctx)
|
|
|
|
vault := earntypes.NewAllowedVault(
|
|
liquidtypes.DefaultDerivativeDenom,
|
|
earntypes.StrategyTypes{earntypes.STRATEGY_TYPE_SAVINGS},
|
|
false,
|
|
nil,
|
|
)
|
|
|
|
earnParams.AllowedVaults = append(earnParams.AllowedVaults, vault)
|
|
ek.SetParams(suite.ctx, earnParams)
|
|
|
|
sk := suite.app.GetSavingsKeeper()
|
|
savingsParams := sk.GetParams(suite.ctx)
|
|
savingsParams.SupportedDenoms = append(savingsParams.SupportedDenoms, liquidtypes.DefaultDerivativeDenom)
|
|
sk.SetParams(suite.ctx, savingsParams)
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) earnDeposit(owner sdk.AccAddress, derivative sdk.Coin) {
|
|
ek := suite.app.GetEarnKeeper()
|
|
|
|
err := ek.Deposit(suite.ctx, owner, derivative, earntypes.STRATEGY_TYPE_SAVINGS)
|
|
suite.Require().NoError(err)
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) mintDerivative(owner sdk.AccAddress, validator sdk.ValAddress, amount sdk.Int) sdk.Coin {
|
|
lk := suite.app.GetLiquidKeeper()
|
|
|
|
minted, err := lk.MintDerivative(suite.ctx, owner, validator, suite.newBondCoin(amount))
|
|
suite.Require().NoError(err)
|
|
|
|
return minted
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) delegateToNewBondedValidator(delegator sdk.AccAddress, amount sdk.Int) stakingtypes.ValidatorI {
|
|
valAcc := suite.createAccount(suite.newBondCoin(sdk.NewInt(1e9)))
|
|
validator, err := suite.staking.createUnbondedValidator(suite.ctx, valAcc.GetAddress().Bytes(), sdk.NewInt(1e9))
|
|
suite.Require().NoError(err)
|
|
|
|
_, err = suite.staking.delegate(suite.ctx, delegator, validator.GetOperator(), amount)
|
|
suite.Require().NoError(err)
|
|
|
|
// bond the validator
|
|
sk := suite.app.GetStakingKeeper()
|
|
staking.EndBlocker(suite.ctx, sk)
|
|
|
|
validator, found := sk.GetValidator(suite.ctx, validator.GetOperator())
|
|
if !found {
|
|
panic("validator not found")
|
|
}
|
|
return validator
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) createNewBondedValidator(selfDelegation sdk.Int) stakingtypes.ValidatorI {
|
|
valAcc := suite.createAccount(suite.newBondCoin(selfDelegation))
|
|
validator, err := suite.staking.createUnbondedValidator(suite.ctx, valAcc.GetAddress().Bytes(), selfDelegation)
|
|
suite.Require().NoError(err)
|
|
|
|
// bond the validator
|
|
sk := suite.app.GetStakingKeeper()
|
|
staking.EndBlocker(suite.ctx, sk)
|
|
|
|
validator, found := sk.GetValidator(suite.ctx, validator.GetOperator())
|
|
if !found {
|
|
panic("validator not found")
|
|
}
|
|
return validator
|
|
}
|
|
|
|
func (suite *tallyHandlerSuite) createAccount(initialBalance ...sdk.Coin) authtypes.AccountI {
|
|
ak := suite.app.GetAccountKeeper()
|
|
|
|
acc := ak.NewAccountWithAddress(suite.ctx, RandomAddress())
|
|
ak.SetAccount(suite.ctx, acc)
|
|
|
|
err := suite.app.FundAccount(suite.ctx, acc.GetAddress(), initialBalance)
|
|
suite.Require().NoError(err)
|
|
|
|
return acc
|
|
}
|
|
|
|
// stakingHelper wraps the staking keeper with helper functions for testing.
|
|
type stakingHelper struct {
|
|
keeper stakingkeeper.Keeper
|
|
}
|
|
|
|
func (h stakingHelper) createUnbondedValidator(ctx sdk.Context, address sdk.ValAddress, selfDelegation sdk.Int) (stakingtypes.ValidatorI, error) {
|
|
msg, err := stakingtypes.NewMsgCreateValidator(
|
|
address,
|
|
ed25519.GenPrivKey().PubKey(),
|
|
h.newBondCoin(ctx, selfDelegation),
|
|
stakingtypes.Description{},
|
|
stakingtypes.NewCommissionRates(sdk.ZeroDec(), sdk.ZeroDec(), sdk.ZeroDec()),
|
|
sdk.NewInt(1e6),
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
msgServer := stakingkeeper.NewMsgServerImpl(h.keeper)
|
|
_, err = msgServer.CreateValidator(sdk.WrapSDKContext(ctx), msg)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
validator, found := h.keeper.GetValidator(ctx, address)
|
|
if !found {
|
|
panic("validator not found")
|
|
}
|
|
return validator, nil
|
|
}
|
|
|
|
func (h stakingHelper) delegate(ctx sdk.Context, delegator sdk.AccAddress, validator sdk.ValAddress, amount sdk.Int) (sdk.Dec, error) {
|
|
msg := stakingtypes.NewMsgDelegate(
|
|
delegator,
|
|
validator,
|
|
h.newBondCoin(ctx, amount),
|
|
)
|
|
|
|
msgServer := stakingkeeper.NewMsgServerImpl(h.keeper)
|
|
_, err := msgServer.Delegate(sdk.WrapSDKContext(ctx), msg)
|
|
if err != nil {
|
|
return sdk.Dec{}, err
|
|
}
|
|
|
|
del, found := h.keeper.GetDelegation(ctx, delegator, validator)
|
|
if !found {
|
|
panic("delegation not found")
|
|
}
|
|
return del.Shares, nil
|
|
}
|
|
|
|
func (h stakingHelper) newBondCoin(ctx sdk.Context, amount sdk.Int) sdk.Coin {
|
|
return sdk.NewCoin(h.keeper.BondDenom(ctx), amount)
|
|
}
|
|
|
|
func (h stakingHelper) setBondDenom(ctx sdk.Context, denom string) {
|
|
params := h.keeper.GetParams(ctx)
|
|
params.BondDenom = denom
|
|
h.keeper.SetParams(ctx, params)
|
|
}
|