From 0e086d61ac701952855543870a5231e59771490e Mon Sep 17 00:00:00 2001 From: Roman Khimov Date: Thu, 20 Aug 2020 19:06:59 +0300 Subject: [PATCH] mempool: store feeSum as big.Int Prevent (very) potential overflow. --- pkg/core/mempool/mem_pool.go | 16 +++++++++------- pkg/core/mempool/mem_pool_test.go | 8 ++++---- 2 files changed, 13 insertions(+), 11 deletions(-) diff --git a/pkg/core/mempool/mem_pool.go b/pkg/core/mempool/mem_pool.go index 521074336..2ccf4001d 100644 --- a/pkg/core/mempool/mem_pool.go +++ b/pkg/core/mempool/mem_pool.go @@ -41,7 +41,7 @@ type items []*item // sender's transactions which are currently in mempool type utilityBalanceAndFees struct { balance *big.Int - feeSum int64 + feeSum *big.Int } // Pool stores the unconfirms transactions. @@ -116,12 +116,13 @@ func (mp *Pool) tryAddSendersFee(tx *transaction.Transaction, feer Feer, needChe senderFee, ok := mp.fees[tx.Sender()] if !ok { senderFee.balance = feer.GetUtilityTokenBalance(tx.Sender()) + senderFee.feeSum = big.NewInt(0) mp.fees[tx.Sender()] = senderFee } if needCheck && checkBalance(tx, senderFee) != nil { return false } - senderFee.feeSum += tx.SystemFee + tx.NetworkFee + senderFee.feeSum.Add(senderFee.feeSum, big.NewInt(tx.SystemFee+tx.NetworkFee)) mp.fees[tx.Sender()] = senderFee return true } @@ -129,12 +130,12 @@ func (mp *Pool) tryAddSendersFee(tx *transaction.Transaction, feer Feer, needChe // checkBalance returns nil in case when sender has enough GAS to pay for the // transaction func checkBalance(tx *transaction.Transaction, balance utilityBalanceAndFees) error { - txFee := tx.SystemFee + tx.NetworkFee - if balance.balance.Cmp(big.NewInt(txFee)) < 0 { + txFee := big.NewInt(tx.SystemFee + tx.NetworkFee) + if balance.balance.Cmp(txFee) < 0 { return ErrInsufficientFunds } - needFee := balance.feeSum + txFee - if balance.balance.Cmp(big.NewInt(needFee)) < 0 { + needFee := txFee.Add(txFee, balance.feeSum) + if balance.balance.Cmp(needFee) < 0 { return ErrConflict } return nil @@ -212,7 +213,7 @@ func (mp *Pool) Remove(hash util.Uint256) { mp.verifiedTxes = mp.verifiedTxes[:num] } senderFee := mp.fees[it.txn.Sender()] - senderFee.feeSum -= it.txn.SystemFee + it.txn.NetworkFee + senderFee.feeSum.Sub(senderFee.feeSum, big.NewInt(it.txn.SystemFee+it.txn.NetworkFee)) mp.fees[it.txn.Sender()] = senderFee } updateMempoolMetrics(len(mp.verifiedTxes)) @@ -299,6 +300,7 @@ func (mp *Pool) checkTxConflicts(tx *transaction.Transaction, fee Feer) error { senderFee, ok := mp.fees[tx.Sender()] if !ok { senderFee.balance = fee.GetUtilityTokenBalance(tx.Sender()) + senderFee.feeSum = big.NewInt(0) } return checkBalance(tx, senderFee) } diff --git a/pkg/core/mempool/mem_pool_test.go b/pkg/core/mempool/mem_pool_test.go index a7420b255..0744f25ef 100644 --- a/pkg/core/mempool/mem_pool_test.go +++ b/pkg/core/mempool/mem_pool_test.go @@ -205,7 +205,7 @@ func TestMemPoolFees(t *testing.T) { require.Equal(t, 1, len(mp.fees)) require.Equal(t, utilityBalanceAndFees{ balance: balance, - feeSum: tx1.NetworkFee, + feeSum: big.NewInt(tx1.NetworkFee), }, mp.fees[sender0]) // balance shouldn't change after adding one more transaction @@ -217,7 +217,7 @@ func TestMemPoolFees(t *testing.T) { require.Equal(t, 1, len(mp.fees)) require.Equal(t, utilityBalanceAndFees{ balance: balance, - feeSum: balance.Int64(), + feeSum: balance, }, mp.fees[sender0]) // can't add more transactions as we don't have enough GAS @@ -229,7 +229,7 @@ func TestMemPoolFees(t *testing.T) { require.Equal(t, 1, len(mp.fees)) require.Equal(t, utilityBalanceAndFees{ balance: balance, - feeSum: balance.Int64(), + feeSum: balance, }, mp.fees[sender0]) // check whether sender's fee updates correctly @@ -242,7 +242,7 @@ func TestMemPoolFees(t *testing.T) { require.Equal(t, 1, len(mp.fees)) require.Equal(t, utilityBalanceAndFees{ balance: balance, - feeSum: tx2.NetworkFee, + feeSum: big.NewInt(tx2.NetworkFee), }, mp.fees[sender0]) // there should be nothing left