Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat!: move Global Fee checks to DeliverTx #2447

Merged
merged 36 commits into from
May 10, 2023
Merged
Show file tree
Hide file tree
Changes from 32 commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
b9a791c
add fee check logic to DeliverTx
sainoe Apr 25, 2023
7b98d13
add uts
sainoe Apr 26, 2023
1a7d9a4
refactor combinedFee
sainoe Apr 26, 2023
35230df
update ut test
sainoe Apr 26, 2023
42e1c55
update uts
sainoe Apr 26, 2023
f969807
save
sainoe Apr 26, 2023
567b74f
print global fee
sainoe Apr 27, 2023
cddadaf
lower fees in e2e tests
sainoe May 1, 2023
79cf97e
fix e2e test tx flags
sainoe May 1, 2023
0bade2c
remove debug logs
sainoe May 1, 2023
a5a29ef
update changelog
sainoe May 1, 2023
238c6fd
nits
sainoe May 1, 2023
e51b618
refactor GetTxFeeRequired
sainoe May 1, 2023
98ab598
nits
sainoe May 1, 2023
016ab54
update docs
sainoe May 1, 2023
c57a4ef
update changelog
sainoe May 1, 2023
a4ed8e1
remove debugging setup
sainoe May 1, 2023
51656cc
nits
sainoe May 1, 2023
bdc180f
nits
sainoe May 1, 2023
0f2a661
nits
sainoe May 1, 2023
7925dfa
nits
sainoe May 1, 2023
d40ac5b
Update x/globalfee/ante/antetest/fee_test.go
sainoe May 2, 2023
b6de6d5
Update x/globalfee/ante/antetest/fee_test.go
sainoe May 2, 2023
83e70f5
fix changelog
sainoe May 3, 2023
22182ee
Merge branch 'main' into sainoe/global-fee-deliver-tx
sainoe May 4, 2023
c3e5dbc
Update x/globalfee/ante/fee.go
sainoe May 5, 2023
00b3335
fix flag fees in e2e tests
sainoe May 8, 2023
6e852fc
Merge branch 'main' into sainoe/global-fee-deliver-tx
sainoe May 8, 2023
4ea3175
refactor: refactor fee check for readability
May 10, 2023
7864d5e
chore: fix typo
May 10, 2023
fd84e3e
chore: update comments
May 10, 2023
a4f34fe
refactor: change map[string]bool to map[string]struct{}
May 10, 2023
3988578
fix linter
sainoe May 10, 2023
398fe54
add key check
May 10, 2023
ec83ca2
remove unnecessary condition and improve test coverage
sainoe May 10, 2023
952d3ff
Update x/globalfee/ante/antetest/fee_test.go
sainoe May 10, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -38,10 +38,11 @@ Ref: https://keepachangelog.com/en/1.0.0/

### Improvements
* (test) [#2440](https://github.com/cosmos/gaia/pull/2440) Add vulncheck to nightly builds
* (gaia) [#2442](https://github.com/cosmos/gaia/pull/2442) Bump [Interchain-Security](https://github.com/cosmos/interchain-security) to [v1.1.1](https://github.com/cosmos/interchain-security/tree/v1.1.1).

### State Machine Breaking
* (gaia) Bump Golang prerequisite from 1.18 to 1.20 see (https://go.dev/blog/go1.20) for details.
* (gaia) [#2442](https://github.com/cosmos/gaia/pull/2442) Bump [Interchain-Security](https://github.com/cosmos/interchain-security) to [v1.1.1](https://github.com/cosmos/interchain-security/tree/v1.1.1).
* (feat!) [#2447] Update Global Fee's AnteHandler to check tx fees against the network min gas prices in DeliverTx mode.
sainoe marked this conversation as resolved.
Show resolved Hide resolved
yaruwangway marked this conversation as resolved.
Show resolved Hide resolved


## [v9.0.3] - 2023-04-19
Expand Down
2 changes: 1 addition & 1 deletion e2e.Dockerfile
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,6 @@ FROM cgr.dev/chainguard/static:$IMG_TAG
ARG IMG_TAG
COPY --from=gaiad-builder /go/bin/gaiad /usr/local/bin/
EXPOSE 26656 26657 1317 9090
USER "nonroot"
USER nonroot

ENTRYPOINT ["gaiad", "start"]
2 changes: 0 additions & 2 deletions go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -1023,8 +1023,6 @@ github.com/openzipkin/zipkin-go v0.1.6/go.mod h1:QgAqvLzwWbR/WpD4A3cGpPtJrZXNIiJ
github.com/openzipkin/zipkin-go v0.2.1/go.mod h1:NaW6tEwdmWMaCDZzg8sh+IBNOxHMPnhQw8ySjnjRyN4=
github.com/openzipkin/zipkin-go v0.2.2/go.mod h1:NaW6tEwdmWMaCDZzg8sh+IBNOxHMPnhQw8ySjnjRyN4=
github.com/ory/dockertest v3.3.5+incompatible h1:iLLK6SQwIhcbrG783Dghaaa3WPzGc+4Emza6EbVUUGA=
github.com/ory/dockertest/v3 v3.9.1 h1:v4dkG+dlu76goxMiTT2j8zV7s4oPPEppKT8K8p2f1kY=
github.com/ory/dockertest/v3 v3.9.1/go.mod h1:42Ir9hmvaAPm0Mgibk6mBPi7SFvTXxEcnztDYOJ//uM=
github.com/ory/dockertest/v3 v3.10.0 h1:4K3z2VMe8Woe++invjaTB7VRyQXQy5UY+loujO4aNE4=
github.com/ory/dockertest/v3 v3.10.0/go.mod h1:nr57ZbRWMqfsdGdFNLHz5jjNdDb7VVFnzAeW1n5N1Lg=
github.com/otiai10/copy v1.2.0/go.mod h1:rrF5dJ5F0t/EWSYODDu4j9/vEeYHMkc8jt0zJChqQWw=
Expand Down
10 changes: 5 additions & 5 deletions tests/e2e/e2e_exec_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -439,8 +439,7 @@ func (s *IntegrationTestSuite) executeDelegate(c *chain, valIdx int, amount, val
amount,
fmt.Sprintf("--%s=%s", flags.FlagFrom, delegatorAddr),
fmt.Sprintf("--%s=%s", flags.FlagChainID, c.id),
fmt.Sprintf("--%s=%s", flags.FlagGas, "auto"),
fmt.Sprintf("--%s=%s", flags.FlagFees, delegateFees),
fmt.Sprintf("--%s=%s", flags.FlagGasPrices, delegateFees),
"--keyring-backend=test",
fmt.Sprintf("--%s=%s", flags.FlagHome, home),
"--output=json",
Expand Down Expand Up @@ -469,8 +468,8 @@ func (s *IntegrationTestSuite) executeRedelegate(c *chain, valIdx int, amount, o
amount,
fmt.Sprintf("--%s=%s", flags.FlagFrom, delegatorAddr),
fmt.Sprintf("--%s=%s", flags.FlagChainID, c.id),
fmt.Sprintf("--%s=%s", flags.FlagGas, "auto"),
fmt.Sprintf("--%s=%s", flags.FlagFees, delegateFees),
fmt.Sprintf("--%s=%s", flags.FlagGas, "300000"), // default 200000 isn't enough
yaruwangway marked this conversation as resolved.
Show resolved Hide resolved
fmt.Sprintf("--%s=%s", flags.FlagGasPrices, delegateFees),
"--keyring-backend=test",
fmt.Sprintf("--%s=%s", flags.FlagHome, home),
"--output=json",
Expand Down Expand Up @@ -538,7 +537,7 @@ func (s *IntegrationTestSuite) execSetWithdrawAddress(
"set-withdraw-addr",
newWithdrawalAddress,
fmt.Sprintf("--%s=%s", flags.FlagFrom, delegatorAddress),
fmt.Sprintf("--%s=%s", flags.FlagGasPrices, fees),
fmt.Sprintf("--%s=%s", flags.FlagFees, fees),
sainoe marked this conversation as resolved.
Show resolved Hide resolved
fmt.Sprintf("--%s=%s", flags.FlagChainID, c.id),
fmt.Sprintf("--%s=%s", flags.FlagHome, homePath),
"--keyring-backend=test",
Expand Down Expand Up @@ -610,6 +609,7 @@ func (s *IntegrationTestSuite) executeGaiaTxCommand(ctx context.Context, c *chai

stdOut := outBuf.Bytes()
stdErr := errBuf.Bytes()

if !validation(stdOut, stdErr) {
s.Require().FailNowf("Exec validation failed", "stdout: %s, stderr: %s",
string(stdOut), string(stdErr))
Expand Down
2 changes: 1 addition & 1 deletion tests/e2e/e2e_staking_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ func (s *IntegrationTestSuite) testStaking() {

delegatorAddress := s.chainA.genesisAccounts[2].keyInfo.GetAddress().String()

fees := sdk.NewCoin(uatomDenom, sdk.NewInt(10))
fees := sdk.NewCoin(uatomDenom, sdk.NewInt(1))

delegationAmount := sdk.NewInt(500000000)
delegation := sdk.NewCoin(uatomDenom, delegationAmount) // 500 atom
Expand Down
3 changes: 2 additions & 1 deletion tests/e2e/e2e_vesting_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import (
"time"

sdk "github.com/cosmos/cosmos-sdk/types"

"github.com/cosmos/gaia/v9/x/globalfee/ante"
)

Expand Down Expand Up @@ -37,7 +38,7 @@ var (
vestingAmount = sdk.NewCoin(uatomDenom, sdk.NewInt(350000))
vestingBalance = sdk.NewCoins(vestingAmountVested).Add(vestingAmount)
vestingDelegationAmount = sdk.NewCoin(uatomDenom, sdk.NewInt(500000000))
vestingDelegationFees = sdk.NewCoin(uatomDenom, sdk.NewInt(10))
vestingDelegationFees = sdk.NewCoin(uatomDenom, sdk.NewInt(1))
)

func (s *IntegrationTestSuite) testDelayedVestingAccount(api string) {
Expand Down
70 changes: 66 additions & 4 deletions x/globalfee/ante/antetest/fee_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import (
cryptotypes "github.com/cosmos/cosmos-sdk/crypto/types"
"github.com/cosmos/cosmos-sdk/testutil/testdata"
sdk "github.com/cosmos/cosmos-sdk/types"
paramtypes "github.com/cosmos/cosmos-sdk/x/params/types"
stakingtypes "github.com/cosmos/cosmos-sdk/x/staking/types"
ibcclienttypes "github.com/cosmos/ibc-go/v4/modules/core/02-client/types"
ibcchanneltypes "github.com/cosmos/ibc-go/v4/modules/core/04-channel/types"
Expand Down Expand Up @@ -574,23 +575,32 @@ func (s *IntegrationTestSuite) TestGlobalFeeMinimumGasFeeAnteHandler() {
txCheck: true,
expErr: true,
},
"disable checkTx: no fee check. min_gas_price is low, global fee is low, tx fee is zero": {
"disable checkTx: min_gas_price is medium, global fee is low, tx fee is low": {
minGasPrice: minGasPrice,
globalFeeParams: globalfeeParamsLow,
gasPrice: sdk.NewCoins(sdk.NewCoin("uatom", sdk.ZeroInt())),
gasPrice: sdk.NewCoins(sdk.NewCoin("uatom", lowFeeAmt)),
gasLimit: testGasLimit,
txMsg: testdata.NewTestMsg(addr1),
txCheck: false,
expErr: false,
},
"disable checkTx: no fee check. min_gas_price is low, global fee is low, tx fee's denom is not in global fees denoms set": {
"disable checkTx: min_gas_price is medium, global fee is low, tx is zero": {
minGasPrice: minGasPrice,
globalFeeParams: globalfeeParamsLow,
gasPrice: sdk.NewCoins(sdk.NewCoin("uatom", sdk.ZeroInt())),
gasLimit: testGasLimit,
txMsg: testdata.NewTestMsg(addr1),
txCheck: false,
expErr: true,
},
"disable checkTx: min_gas_price is low, global fee is low, tx fee's denom is not in global fees denoms set": {
minGasPrice: minGasPrice,
globalFeeParams: globalfeeParamsLow,
gasPrice: sdk.NewCoins(sdk.NewCoin("quark", sdk.ZeroInt())),
gasLimit: testGasLimit,
txMsg: testdata.NewTestMsg(addr1),
txCheck: false,
expErr: false,
expErr: true,
},
}
for name, tc := range testCases {
Expand Down Expand Up @@ -746,3 +756,55 @@ func (s *IntegrationTestSuite) TestContainsOnlyBypassMinFeeMsgs() {
})
}
}

func (s *IntegrationTestSuite) TestGetTxFeeRequired() {
// create global fee params
globalfeeParamsEmpty := &globfeetypes.Params{MinimumGasPrices: []sdk.DecCoin{}}

// setup tests with default global fee i.e. "0uatom" and empty local min gas prices
feeDecorator, _ := s.SetupTestGlobalFeeStoreAndMinGasPrice([]sdk.DecCoin{}, globalfeeParamsEmpty)

// reset decorator staking subspace
feeDecorator.StakingSubspace = paramtypes.Subspace{}

// check that an error is returned when staking subspace isn't set
sainoe marked this conversation as resolved.
Show resolved Hide resolved
_, err := feeDecorator.GetTxFeeRequired(s.ctx, nil)
s.Require().Equal(err.Error(), "empty staking bond denomination")

// set non-zero local min gas prices
localMinGasPrices := sdk.NewCoins(sdk.NewCoin("uatom", sdk.NewInt(1)))

// setup tests with non-empty local min gas prices
feeDecorator, _ = s.SetupTestGlobalFeeStoreAndMinGasPrice(
sdk.NewDecCoinsFromCoins(localMinGasPrices...),
globalfeeParamsEmpty,
)

// mock tx data
s.txBuilder = s.clientCtx.TxConfig.NewTxBuilder()
priv1, _, addr1 := testdata.KeyTestPubAddr()
privs, accNums, accSeqs := []cryptotypes.PrivKey{priv1}, []uint64{0}, []uint64{0}

s.Require().NoError(s.txBuilder.SetMsgs(testdata.NewTestMsg(addr1)))
s.txBuilder.SetFeeAmount(sdk.NewCoins(sdk.NewCoin("uatom", sdk.ZeroInt())))

s.txBuilder.SetGasLimit(uint64(1))
tx, err := s.CreateTestTx(privs, accNums, accSeqs, s.ctx.ChainID())
s.Require().NoError(err)

// check that the required fees returned in CheckTx mode are equal to
// local min gas prices since they're greater than the default global fee values.
s.Require().True(s.ctx.IsCheckTx())
res, err := feeDecorator.GetTxFeeRequired(s.ctx, tx)
s.Require().True(res.IsEqual(localMinGasPrices))
s.Require().NoError(err)

// check that the global fee is returned in DeliverTx mode.
globalFee, err := feeDecorator.GetGlobalFee(s.ctx, tx)
s.Require().NoError(err)

ctx := s.ctx.WithIsCheckTx(false)
res, err = feeDecorator.GetTxFeeRequired(ctx, tx)
s.Require().NoError(err)
s.Require().True(res.IsEqual(globalFee))
}
141 changes: 87 additions & 54 deletions x/globalfee/ante/fee.go
Original file line number Diff line number Diff line change
Expand Up @@ -57,93 +57,121 @@ func (mfd FeeDecorator) AnteHandle(ctx sdk.Context, tx sdk.Tx, simulate bool, ne
return ctx, sdkerrors.Wrap(sdkerrors.ErrTxDecode, "Tx must implement the sdk.FeeTx interface")
}

// Only check for minimum fees and global fee if the execution mode is CheckTx
if !ctx.IsCheckTx() || simulate {
// Do not check minimum-gas-prices and global fees during simulations
if simulate {
return next(ctx, tx, simulate)
}

// Sort fee tx's coins, zero coins in feeCoins are already removed
feeCoins := feeTx.GetFee().Sort()
gas := feeTx.GetGas()
msgs := feeTx.GetMsgs()

// Get required Global Fee
requiredGlobalFees, err := mfd.GetGlobalFee(ctx, feeTx)
// Get the required fees according to the CheckTx or DeliverTx modes
feeRequired, err := mfd.GetTxFeeRequired(ctx, feeTx)
if err != nil {
return ctx, err
}

// reject the transaction early if the feeCoins have more denoms than the fee requirement
if feeCoins.Len() > requiredGlobalFees.Len() {
return ctx, sdkerrors.Wrapf(sdkerrors.ErrInvalidCoins, "fee is not a subset of required fees; got %s, required: %s", feeCoins.String(), requiredGlobalFees.String())
// feeRequired cannot be be empty
if feeTx.GetFee().Len() > feeRequired.Len() {
return ctx, sdkerrors.Wrapf(sdkerrors.ErrInvalidCoins, "fee is not a subset of required fees; got %s, required: %s", feeTx.GetFee().String(), feeRequired.String())
}

// Get local minimum-gas-prices
localFees := GetMinGasPrice(ctx, int64(feeTx.GetGas()))

// CombinedFeeRequirement should never be empty since
// global fee is set to its default value, i.e. 0uatom, if empty
combinedFeeRequirement := CombinedFeeRequirement(requiredGlobalFees, localFees)
if len(combinedFeeRequirement) == 0 {
return ctx, sdkerrors.Wrapf(sdkerrors.ErrNotFound, "required fees are not setup.")
}
// Sort fee tx's coins, zero coins in feeCoins are already removed
feeCoins := feeTx.GetFee().Sort()
gas := feeTx.GetGas()
msgs := feeTx.GetMsgs()

nonZeroCoinFeesReq, zeroCoinFeesDenomReq := getNonZeroFees(combinedFeeRequirement)
// split feeRequired into zero and non-zero coins(nonZeroCoinFeesReq, zeroCoinFeesDenomReq), split feeCoins according to
// nonZeroCoinFeesReq, zeroCoinFeesDenomReq,
// so that feeCoins can be checked separately against them.
nonZeroCoinFeesReq, zeroCoinFeesDenomReq := getNonZeroFees(feeRequired)
yaruwangway marked this conversation as resolved.
Show resolved Hide resolved

// feeCoinsNonZeroDenom contains non-zero denominations from the combinedFeeRequirement
//
// feeCoinsNoZeroDenom is used to check if the fees meets the requirement imposed by nonZeroCoinFeesReq
// when feeCoins does not contain zero coins' denoms in combinedFeeRequirement
// feeCoinsNonZeroDenom contains non-zero denominations from the feeRequired
// feeCoinsNonZeroDenom is used to check if the fees meets the requirement imposed by nonZeroCoinFeesReq
// when feeCoins does not contain zero coins' denoms in feeRequired
feeCoinsNonZeroDenom, feeCoinsZeroDenom := splitCoinsByDenoms(feeCoins, zeroCoinFeesDenomReq)

// Check that the fees are in expected denominations.
// if feeCoinsNoZeroDenom=[], DenomsSubsetOf returns true
// if feeCoinsNoZeroDenom is not empty, but nonZeroCoinFeesReq empty, return false
// according to splitCoinsByDenoms(), feeCoinsZeroDenom must be in denom subset of zeroCoinFeesDenomReq.
// check if feeCoinsNonZeroDenom is a subset of nonZeroCoinFeesReq.
// special case: if feeCoinsNonZeroDenom=[], DenomsSubsetOf returns true
// special case: if feeCoinsNonZeroDenom is not empty, but nonZeroCoinFeesReq empty, return false
if !feeCoinsNonZeroDenom.DenomsSubsetOf(nonZeroCoinFeesReq) {
return ctx, sdkerrors.Wrapf(sdkerrors.ErrInvalidCoins, "fee is not a subset of required fees; got %s, required: %s", feeCoins.String(), combinedFeeRequirement.String())
return ctx, sdkerrors.Wrapf(sdkerrors.ErrInsufficientFee, "fee is not a subset of required fees; got %s, required: %s", feeCoins.String(), feeRequired.String())
}

// Accept zero fee transactions only if both of the following statements are true:
// If the feeCoins pass the denoms check, check they are bypass-msg types.
//
// Bypass min fee requires:
// - the tx contains only message types that can bypass the minimum fee,
// see BypassMinFeeMsgTypes;
// - the total gas limit per message does not exceed MaxTotalBypassMinFeeMsgGasUsage,
// i.e., totalGas <= MaxTotalBypassMinFeeMsgGasUsage
//
// Otherwise, minimum fees and global fees are checked to prevent spam.
doesNotExceedMaxGasUsage := gas <= mfd.MaxTotalBypassMinFeeMsgGasUsage
allowedToBypassMinFee := mfd.ContainsOnlyBypassMinFeeMsgs(msgs) && doesNotExceedMaxGasUsage
if allowedToBypassMinFee {
return next(ctx, tx, simulate)
}

// Either the transaction contains at least one message of a type
// that cannot bypass the minimum fee or the total gas limit exceeds
// the imposed threshold. As a result, besides check the fees are in
// expected denominations, check the amounts are greater or equal than
// the expected amounts.

// only check feeCoinsNoZeroDenom has coins IsAnyGTE than nonZeroCoinFeesReq
// when feeCoins does not contain denoms of zero denoms in combinedFeeRequirement
if !allowedToBypassMinFee && len(feeCoinsZeroDenom) == 0 {
// special case: when feeCoins=[] and there is zero coin in fee requirement
if len(feeCoins) == 0 && len(zeroCoinFeesDenomReq) != 0 {
// if the msg does not satisfy bypass condition and the feeCoins denoms are subset of feeRequired,
// check the feeCoins amount against feeRequired
//
// when feeCoins=[]
// special case: and there is zero coin in fee requirement, pass,
// otherwise, err
if len(feeCoins) == 0 {
if len(zeroCoinFeesDenomReq) != 0 {
return next(ctx, tx, simulate)
} else {
return ctx, sdkerrors.Wrapf(sdkerrors.ErrInsufficientFee, "insufficient fees; got: %s required: %s", feeCoins.String(), feeRequired.String())
}
}

// Check that the amounts of the fees are greater or equal than
// the expected amounts, i.e., at least one feeCoin amount must
// be greater or equal to one of the combined required fees.
// when feeCoins != []
// special case: if TX has at least one of the zeroCoinFeesDenomReq, then it should pass
if len(feeCoinsZeroDenom) > 0 {
return next(ctx, tx, simulate)
}

// if feeCoinsNoZeroDenom=[], return false
// if nonZeroCoinFeesReq=[], return false (this situation should not happen
// because when nonZeroCoinFeesReq empty, and DenomsSubsetOf check passed,
// the tx should already passed before)
if !feeCoinsNonZeroDenom.IsAnyGTE(nonZeroCoinFeesReq) {
return ctx, sdkerrors.Wrapf(sdkerrors.ErrInsufficientFee, "insufficient fees; got: %s required: %s", feeCoins.String(), combinedFeeRequirement.String())
}
// After all the checks, the tx is confirmed:
// feeCoins denoms subset off feeRequired
// Not bypass
// feeCoins != []
// Not contain zeroCoinFeesDenomReq's denoms
//
// check if the feeCoins's feeCoinsNonZeroDenom part has coins' amount higher/equal to nonZeroCoinFeesReq
if !feeCoinsNonZeroDenom.IsAnyGTE(nonZeroCoinFeesReq) {
return ctx, sdkerrors.Wrapf(sdkerrors.ErrInsufficientFee, "insufficient fees; got: %s required: %s", feeCoins.String(), feeRequired.String())
}

return next(ctx, tx, simulate)
}

// GetTxFeeRequired returns the required fees for the given FeeTx.
// In case the FeeTx's mode is CheckTx, it returns the combined requirements
// of local min gas prices and global fees. Otherwise, in DeliverTx, it returns the global fee.
func (mfd FeeDecorator) GetTxFeeRequired(ctx sdk.Context, tx sdk.FeeTx) (sdk.Coins, error) {
// Get required global fee min gas prices
// Note that it should never be empty since its default value is set to coin={"StakingBondDenom", 0}
globalFees, err := mfd.GetGlobalFee(ctx, tx)
if err != nil {
return sdk.Coins{}, err
}

// In DeliverTx, the global fee min gas prices are the only tx fee requirements.
if !ctx.IsCheckTx() {
sainoe marked this conversation as resolved.
Show resolved Hide resolved
return globalFees, nil
}

// In CheckTx mode, the local and global fee min gas prices are combined
// to form the tx fee requirements

// Get local minimum-gas-prices
localFees := GetMinGasPrice(ctx, int64(tx.GetGas()))

// Return combined fee requirements
return CombinedFeeRequirement(globalFees, localFees)
}

// GetGlobalFee returns the global fees for a given fee tx's gas
// (might also return 0denom if globalMinGasPrice is 0)
// sorted in ascending order.
Expand Down Expand Up @@ -176,6 +204,7 @@ func (mfd FeeDecorator) GetGlobalFee(ctx sdk.Context, feeTx sdk.FeeTx) (sdk.Coin
return requiredGlobalFees.Sort(), nil
}

// DefaultZeroGlobalFee returns a zero coin with the staking module bond denom
func (mfd FeeDecorator) DefaultZeroGlobalFee(ctx sdk.Context) ([]sdk.DecCoin, error) {
bondDenom := mfd.getBondDenom(ctx)
if bondDenom == "" {
Expand All @@ -186,11 +215,15 @@ func (mfd FeeDecorator) DefaultZeroGlobalFee(ctx sdk.Context) ([]sdk.DecCoin, er
}

func (mfd FeeDecorator) getBondDenom(ctx sdk.Context) string {
var bondDenom string
if mfd.StakingSubspace.Has(ctx, stakingtypes.KeyBondDenom) {
mfd.StakingSubspace.Get(ctx, stakingtypes.KeyBondDenom, &bondDenom)
// prevent the getter above to panic
// when the staking subspace isn't set
yaruwangway marked this conversation as resolved.
Show resolved Hide resolved
if !mfd.StakingSubspace.HasKeyTable() {
yaruwangway marked this conversation as resolved.
Show resolved Hide resolved
return ""
}

var bondDenom string
mfd.StakingSubspace.Get(ctx, stakingtypes.KeyBondDenom, &bondDenom)
yaruwangway marked this conversation as resolved.
Show resolved Hide resolved

return bondDenom
}

Expand Down
Loading