Multiple fees in min fees to OR instead of AND (#3239)

This commit is contained in:
Zaki Manian 2019-01-08 08:21:54 -08:00 committed by Jack Zampolin
parent 74aa9844fc
commit ed2b6bd9a7
4 changed files with 103 additions and 4 deletions

View File

@ -40,7 +40,7 @@ func TestGaiaCLIMinimumFees(t *testing.T) {
flags := fmt.Sprintf("--home=%s --node=%v --chain-id=%v", gaiacliHome, servAddr, chainID)
// start gaiad server with minimum fees
proc := tests.GoExecuteTWithStdout(t, fmt.Sprintf("gaiad start --home=%s --rpc.laddr=%v --p2p.laddr=%v --minimum_fees=2feetoken", gaiadHome, servAddr, p2pAddr))
proc := tests.GoExecuteTWithStdout(t, fmt.Sprintf("gaiad start --home=%s --rpc.laddr=%v --p2p.laddr=%v --minimum_fees=2%s,2feetoken", gaiadHome, servAddr, p2pAddr, stakeTypes.DefaultBondDenom))
defer proc.Stop(false)
tests.WaitForTMStart(port)
@ -52,9 +52,28 @@ func TestGaiaCLIMinimumFees(t *testing.T) {
fooAcc := executeGetAccount(t, fmt.Sprintf("gaiacli query account %s %v", fooAddr, flags))
require.Equal(t, int64(50), fooAcc.GetCoins().AmountOf(stakeTypes.DefaultBondDenom).Int64())
// Ensure that a tx with no fees fails
success := executeWrite(t, fmt.Sprintf(
"gaiacli tx send %v --amount=10%s --to=%s --from=foo", flags, stakeTypes.DefaultBondDenom, barAddr), app.DefaultKeyPass)
require.False(t, success)
// Ensure tx with correct fees (stake) pass
success = executeWrite(t, fmt.Sprintf(
"gaiacli tx send %v --amount=10%s --to=%s --from=foo --fees=23%s", flags, stakeTypes.DefaultBondDenom, barAddr, stakeTypes.DefaultBondDenom), app.DefaultKeyPass)
require.True(t, success)
tests.WaitForNextNBlocksTM(1, port)
// Ensure tx with correct fees (feetoken) pass
success = executeWrite(t, fmt.Sprintf(
"gaiacli tx send %v --amount=10feetoken --to=%s --from=foo --fees=23feetoken", flags, barAddr), app.DefaultKeyPass)
require.True(t, success)
tests.WaitForNextNBlocksTM(1, port)
// Ensure tx with improper fees fails
success = executeWrite(t, fmt.Sprintf(
"gaiacli tx send %v --amount=10%s --to=%s --from=foo --fees=2footoken", flags, stakeTypes.DefaultBondDenom, barAddr), app.DefaultKeyPass)
require.False(t, success)
cleanupDirs(gaiadHome, gaiacliHome)
}
@ -732,7 +751,7 @@ func initializeFixtures(t *testing.T) (chainID, servAddr, port, gaiadHome, gaiac
chainID = executeInit(t, fmt.Sprintf("gaiad init -o --moniker=foo --home=%s", gaiadHome))
executeWriteCheckErr(t, fmt.Sprintf(
"gaiad add-genesis-account %s 150%s,1000footoken --home=%s", fooAddr, stakeTypes.DefaultBondDenom, gaiadHome))
"gaiad add-genesis-account %s 150%s,1000footoken,1000feetoken --home=%s", fooAddr, stakeTypes.DefaultBondDenom, gaiadHome))
executeWrite(t, fmt.Sprintf("cat %s%sconfig%sgenesis.json", gaiadHome, string(os.PathSeparator), string(os.PathSeparator)))
executeWriteCheckErr(t, fmt.Sprintf(
"gaiad gentx --name=foo --home=%s --home-client=%s", gaiadHome, gaiacliHome), app.DefaultKeyPass)

View File

@ -265,7 +265,29 @@ func (coins Coins) SafeMinus(coinsB Coins) (Coins, bool) {
return diff, !diff.IsNotNegative()
}
// IsAllGT returns true iff for every denom in coins, the denom is present at a
// IsAnyGT returns true if coins contains at least one denom
// that is present at a smaller amount in coinsB; it
// returns false otherwise.
func (coins Coins) IsAnyGT(coinsB Coins) bool {
intersection := coins.intersect(coinsB)
if intersection.Empty() {
return false
}
diff, _ := intersection.SafeMinus(coinsB)
if len(diff) == 0 {
return false
}
for _, coin := range diff {
if coin.IsPositive() {
return true
}
}
return false
}
// IsAllGT returns true if for every denom in coins, the denom is present at a
// greater amount in coinsB.
func (coins Coins) IsAllGT(coinsB Coins) bool {
diff, _ := coins.SafeMinus(coinsB)
@ -276,6 +298,28 @@ func (coins Coins) IsAllGT(coinsB Coins) bool {
return diff.IsPositive()
}
// IsAnyGT returns true if coins contains at least one denom
// that is present at a smaller or equal amount in coinsB; it
// returns false otherwise.
func (coins Coins) IsAnyGTE(coinsB Coins) bool {
intersection := coins.intersect(coinsB)
if intersection.Empty() {
return false
}
diff, _ := intersection.SafeMinus(coinsB)
if len(diff) == 0 || len(diff) < len(intersection) { // zero diff is removed from the diff set
return true
}
for _, coin := range diff {
if coin.IsNotNegative() {
return true
}
}
return false
}
// IsAllGTE returns true iff for every denom in coins, the denom is present at
// an equal or greater amount in coinsB.
func (coins Coins) IsAllGTE(coinsB Coins) bool {
@ -430,6 +474,20 @@ func removeZeroCoins(coins Coins) Coins {
return coins[:i]
}
func (coins Coins) intersect(coinsB Coins) Coins {
intersection := Coins{}
for _, coin := range coins {
for _, bCoin := range coinsB {
if coin.Denom == bCoin.Denom {
intersection = append(intersection, coin)
break
}
}
}
return intersection
}
//-----------------------------------------------------------------------------
// Sort interface

View File

@ -368,6 +368,28 @@ func TestCoinsLTE(t *testing.T) {
assert.True(t, Coins{}.IsAllLTE(Coins{{"a", one}}))
}
func TestCoinsIsAnyGT(t *testing.T) {
one := NewInt(1)
two := NewInt(2)
assert.False(t, Coins{}.IsAnyGT(Coins{}))
assert.False(t, Coins{{"a", one}}.IsAnyGT(Coins{}))
assert.False(t, Coins{}.IsAnyGT(Coins{{"a", one}}))
assert.False(t, Coins{{"a", one}}.IsAnyGT(Coins{{"a", one}}))
assert.True(t, Coins{{"a", one}, {"b", two}}.IsAnyGT(Coins{{"a", one}, {"b", one}}))
assert.True(t, Coins{{"a", two}, {"b", one}}.IsAnyGT(Coins{{"a", one}, {"b", two}}))
}
func TestCoinsIsAnyGTE(t *testing.T) {
one := NewInt(1)
two := NewInt(2)
assert.False(t, Coins{}.IsAnyGTE(Coins{}))
assert.False(t, Coins{{"a", one}}.IsAnyGTE(Coins{}))
assert.False(t, Coins{}.IsAnyGTE(Coins{{"a", one}}))
assert.True(t, Coins{{"a", one}}.IsAnyGTE(Coins{{"a", one}}))
assert.True(t, Coins{{"a", one}, {"b", two}}.IsAnyGTE(Coins{{"a", one}, {"b", one}}))
assert.True(t, Coins{{"a", one}, {"b", one}}.IsAnyGTE(Coins{{"a", one}, {"b", two}}))
}
func TestParse(t *testing.T) {
one := NewInt(1)

View File

@ -294,7 +294,7 @@ func EnsureSufficientMempoolFees(ctx sdk.Context, stdTx StdTx) sdk.Result {
requiredFees := adjustFeesByGas(ctx.MinimumFees(), stdTx.Fee.Gas)
// NOTE: !A.IsAllGTE(B) is not the same as A.IsAllLT(B).
if !ctx.MinimumFees().IsZero() && !stdTx.Fee.Amount.IsAllGTE(requiredFees) {
if !ctx.MinimumFees().IsZero() && !stdTx.Fee.Amount.IsAnyGTE(requiredFees) {
// validators reject any tx from the mempool with less than the minimum fee per gas * gas factor
return sdk.ErrInsufficientFee(
fmt.Sprintf(