From 4be1009def8473cdabb4d897101c7fb74cce8ce0 Mon Sep 17 00:00:00 2001 From: Anna Shaleva Date: Thu, 9 Jul 2020 18:27:20 +0300 Subject: [PATCH 1/2] core: refactor checkBalanceAndUpdate --- pkg/core/mempool/mem_pool.go | 13 ++++++++----- 1 file changed, 8 insertions(+), 5 deletions(-) diff --git a/pkg/core/mempool/mem_pool.go b/pkg/core/mempool/mem_pool.go index 72703b9c4..9c9a006c3 100644 --- a/pkg/core/mempool/mem_pool.go +++ b/pkg/core/mempool/mem_pool.go @@ -124,11 +124,14 @@ func (mp *Pool) checkBalanceAndUpdate(tx *transaction.Transaction, feer Feer) bo senderFee.balance = feer.GetUtilityTokenBalance(tx.Sender) mp.fees[tx.Sender] = senderFee } - needFee := senderFee.feeSum + tx.SystemFee + tx.NetworkFee - if senderFee.balance.Cmp(big.NewInt(needFee)) < 0 { - return false - } - return true + return checkBalance(tx, senderFee) +} + +// checkBalance returns true in case when sender has enough GAS to pay for the +// transaction +func checkBalance(tx *transaction.Transaction, balance utilityBalanceAndFees) bool { + needFee := balance.feeSum + tx.SystemFee + tx.NetworkFee + return balance.balance.Cmp(big.NewInt(needFee)) >= 0 } // addSendersFee adds system fee and network fee to the total sender`s fee in mempool From 789ee2d3c16444e86435719b073c2582c1a93a9d Mon Sep 17 00:00:00 2001 From: Anna Shaleva Date: Thu, 9 Jul 2020 19:23:41 +0300 Subject: [PATCH 2/2] core: do not update mempool while verifying tx Closes #1168 --- pkg/core/mempool/mem_pool.go | 39 +++++++++++++------------------ pkg/core/mempool/mem_pool_test.go | 8 ++----- 2 files changed, 18 insertions(+), 29 deletions(-) diff --git a/pkg/core/mempool/mem_pool.go b/pkg/core/mempool/mem_pool.go index 9c9a006c3..c6777c536 100644 --- a/pkg/core/mempool/mem_pool.go +++ b/pkg/core/mempool/mem_pool.go @@ -107,24 +107,19 @@ func (mp *Pool) containsKey(hash util.Uint256) bool { } // tryAddSendersFee tries to add system fee and network fee to the total sender`s fee in mempool -// and returns false if sender has not enough GAS to pay -func (mp *Pool) tryAddSendersFee(tx *transaction.Transaction, feer Feer) bool { - if !mp.checkBalanceAndUpdate(tx, feer) { - return false - } - mp.addSendersFee(tx) - return true -} - -// checkBalanceAndUpdate returns true in case when sender has enough GAS to pay for -// the transaction and sets sender's balance value in mempool in case if it was not set -func (mp *Pool) checkBalanceAndUpdate(tx *transaction.Transaction, feer Feer) bool { +// and returns false if both balance check is required and sender has not enough GAS to pay +func (mp *Pool) tryAddSendersFee(tx *transaction.Transaction, feer Feer, needCheck bool) bool { senderFee, ok := mp.fees[tx.Sender] if !ok { senderFee.balance = feer.GetUtilityTokenBalance(tx.Sender) mp.fees[tx.Sender] = senderFee } - return checkBalance(tx, senderFee) + if needCheck && !checkBalance(tx, senderFee) { + return false + } + senderFee.feeSum += tx.SystemFee + tx.NetworkFee + mp.fees[tx.Sender] = senderFee + return true } // checkBalance returns true in case when sender has enough GAS to pay for the @@ -134,13 +129,6 @@ func checkBalance(tx *transaction.Transaction, balance utilityBalanceAndFees) bo return balance.balance.Cmp(big.NewInt(needFee)) >= 0 } -// addSendersFee adds system fee and network fee to the total sender`s fee in mempool -func (mp *Pool) addSendersFee(tx *transaction.Transaction) { - senderFee := mp.fees[tx.Sender] - senderFee.feeSum += tx.SystemFee + tx.NetworkFee - mp.fees[tx.Sender] = senderFee -} - // Add tries to add given transaction to the Pool. func (mp *Pool) Add(t *transaction.Transaction, fee Feer) error { var pItem = &item{ @@ -186,7 +174,8 @@ func (mp *Pool) Add(t *transaction.Transaction, fee Feer) error { copy(mp.verifiedTxes[n+1:], mp.verifiedTxes[n:]) mp.verifiedTxes[n] = pItem } - mp.addSendersFee(pItem.txn) + // we already checked balance in checkTxConflicts, so don't need to check again + mp.tryAddSendersFee(pItem.txn, fee, false) updateMempoolMetrics(len(mp.verifiedTxes)) mp.lock.Unlock() @@ -229,7 +218,7 @@ func (mp *Pool) RemoveStale(isOK func(*transaction.Transaction) bool, feer Feer) newVerifiedTxes := mp.verifiedTxes[:0] mp.fees = make(map[util.Uint160]utilityBalanceAndFees) // it'd be nice to reuse existing map, but we can't easily clear it for _, itm := range mp.verifiedTxes { - if isOK(itm.txn) && mp.checkPolicy(itm.txn, policyChanged) && mp.tryAddSendersFee(itm.txn, feer) { + if isOK(itm.txn) && mp.checkPolicy(itm.txn, policyChanged) && mp.tryAddSendersFee(itm.txn, feer, true) { newVerifiedTxes = append(newVerifiedTxes, itm) } else { delete(mp.verifiedMap, itm.txn.Hash()) @@ -295,7 +284,11 @@ func (mp *Pool) GetVerifiedTransactions() []*transaction.Transaction { // checkTxConflicts is an internal unprotected version of Verify. func (mp *Pool) checkTxConflicts(tx *transaction.Transaction, fee Feer) bool { - return mp.checkBalanceAndUpdate(tx, fee) + senderFee, ok := mp.fees[tx.Sender] + if !ok { + senderFee.balance = fee.GetUtilityTokenBalance(tx.Sender) + } + return checkBalance(tx, senderFee) } // Verify checks if a Sender of tx is able to pay for it (and all the other diff --git a/pkg/core/mempool/mem_pool_test.go b/pkg/core/mempool/mem_pool_test.go index c3c5d5a15..d1df45bcf 100644 --- a/pkg/core/mempool/mem_pool_test.go +++ b/pkg/core/mempool/mem_pool_test.go @@ -187,14 +187,10 @@ func TestMemPoolFees(t *testing.T) { tx0 := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0) tx0.NetworkFee = balance.Int64() + 1 tx0.Sender = sender0 - // insufficient funds to add transaction, but balance should be stored + // insufficient funds to add transaction, and balance shouldn't be stored require.Equal(t, false, mp.Verify(tx0, &FeerStub{})) require.Error(t, mp.Add(tx0, &FeerStub{})) - require.Equal(t, 1, len(mp.fees)) - require.Equal(t, utilityBalanceAndFees{ - balance: balance, - feeSum: 0, - }, mp.fees[sender0]) + require.Equal(t, 0, len(mp.fees)) balancePart := new(big.Int).Div(balance, big.NewInt(4)) // no problems with adding another transaction with lower fee