core: don't search through the whole DAO in isTxStillRelevant
New transactions are added to the chain with blocks. If there is no transaction X at height N in DAO, it could only be added with block N+1, so it has to be present there. Therefore we can replace `dao.HasTransaction()` check with a search through in-block transactions. HasTransaction() is nasty in that it may add useless load the DB and this code is being run with a big Blockchain lock held, so we don't want to be delayed here at all. Improves single-node TPS by ~2%.
This commit is contained in:
parent
3a379e4f3e
commit
c7032022f8
2 changed files with 46 additions and 10 deletions
|
@ -4,6 +4,7 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math/big"
|
"math/big"
|
||||||
|
"sort"
|
||||||
"sync"
|
"sync"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
"time"
|
"time"
|
||||||
|
@ -607,7 +608,8 @@ func (bc *Blockchain) storeBlock(block *block.Block) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tx := range block.Transactions {
|
var txHashes = make([]util.Uint256, len(block.Transactions))
|
||||||
|
for i, tx := range block.Transactions {
|
||||||
if err := cache.StoreAsTransaction(tx, block.Index); err != nil {
|
if err := cache.StoreAsTransaction(tx, block.Index); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -624,8 +626,8 @@ func (bc *Blockchain) storeBlock(block *block.Block) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("failed to persist invocation results: %w", err)
|
return fmt.Errorf("failed to persist invocation results: %w", err)
|
||||||
}
|
}
|
||||||
for i := range systemInterop.Notifications {
|
for j := range systemInterop.Notifications {
|
||||||
bc.handleNotification(&systemInterop.Notifications[i], cache, block, tx.Hash())
|
bc.handleNotification(&systemInterop.Notifications[j], cache, block, tx.Hash())
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
bc.log.Warn("contract invocation failed",
|
bc.log.Warn("contract invocation failed",
|
||||||
|
@ -646,7 +648,11 @@ func (bc *Blockchain) storeBlock(block *block.Block) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("failed to store tx exec result: %w", err)
|
return fmt.Errorf("failed to store tx exec result: %w", err)
|
||||||
}
|
}
|
||||||
|
txHashes[i] = tx.Hash()
|
||||||
}
|
}
|
||||||
|
sort.Slice(txHashes, func(i, j int) bool {
|
||||||
|
return txHashes[i].CompareTo(txHashes[j]) < 0
|
||||||
|
})
|
||||||
|
|
||||||
root := bc.dao.MPT.StateRoot()
|
root := bc.dao.MPT.StateRoot()
|
||||||
var prevHash util.Uint256
|
var prevHash util.Uint256
|
||||||
|
@ -688,7 +694,7 @@ func (bc *Blockchain) storeBlock(block *block.Block) error {
|
||||||
}
|
}
|
||||||
bc.topBlock.Store(block)
|
bc.topBlock.Store(block)
|
||||||
atomic.StoreUint32(&bc.blockHeight, block.Index)
|
atomic.StoreUint32(&bc.blockHeight, block.Index)
|
||||||
bc.memPool.RemoveStale(bc.isTxStillRelevant, bc)
|
bc.memPool.RemoveStale(func(tx *transaction.Transaction) bool { return bc.isTxStillRelevant(tx, txHashes) }, bc)
|
||||||
bc.lock.Unlock()
|
bc.lock.Unlock()
|
||||||
|
|
||||||
updateBlockHeightMetric(block.Index)
|
updateBlockHeightMetric(block.Index)
|
||||||
|
@ -1251,16 +1257,18 @@ func (bc *Blockchain) verifyTx(t *transaction.Transaction, block *block.Block) e
|
||||||
}
|
}
|
||||||
|
|
||||||
// isTxStillRelevant is a callback for mempool transaction filtering after the
|
// isTxStillRelevant is a callback for mempool transaction filtering after the
|
||||||
// new block addition. It returns false for transactions already present in the
|
// new block addition. It returns false for transactions added by the new block
|
||||||
// chain (added by the new block), transactions using some inputs that are
|
// (passed via txHashes) and does witness reverification for non-standard
|
||||||
// already used (double spends) and does witness reverification for non-standard
|
|
||||||
// contracts. It operates under the assumption that full transaction verification
|
// contracts. It operates under the assumption that full transaction verification
|
||||||
// was already done so we don't need to check basic things like size, input/output
|
// was already done so we don't need to check basic things like size, input/output
|
||||||
// correctness, etc.
|
// correctness, presence in blocks before the new one, etc.
|
||||||
func (bc *Blockchain) isTxStillRelevant(t *transaction.Transaction) bool {
|
func (bc *Blockchain) isTxStillRelevant(t *transaction.Transaction, txHashes []util.Uint256) bool {
|
||||||
var recheckWitness bool
|
var recheckWitness bool
|
||||||
|
|
||||||
if bc.dao.HasTransaction(t.Hash()) {
|
index := sort.Search(len(txHashes), func(i int) bool {
|
||||||
|
return txHashes[i].CompareTo(t.Hash()) >= 0
|
||||||
|
})
|
||||||
|
if index < len(txHashes) && txHashes[index].Equals(t.Hash()) {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
for i := range t.Scripts {
|
for i := range t.Scripts {
|
||||||
|
|
|
@ -402,6 +402,34 @@ func TestVerifyHashAgainstScript(t *testing.T) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestMemPoolRemoval(t *testing.T) {
|
||||||
|
const added = 16
|
||||||
|
const notAdded = 32
|
||||||
|
bc := newTestChain(t)
|
||||||
|
defer bc.Close()
|
||||||
|
addedTxes := make([]*transaction.Transaction, added)
|
||||||
|
notAddedTxes := make([]*transaction.Transaction, notAdded)
|
||||||
|
for i := range addedTxes {
|
||||||
|
addedTxes[i] = bc.newTestTx(testchain.MultisigScriptHash(), []byte{byte(opcode.PUSH1)})
|
||||||
|
require.NoError(t, signTx(bc, addedTxes[i]))
|
||||||
|
require.NoError(t, bc.PoolTx(addedTxes[i]))
|
||||||
|
}
|
||||||
|
for i := range notAddedTxes {
|
||||||
|
notAddedTxes[i] = bc.newTestTx(testchain.MultisigScriptHash(), []byte{byte(opcode.PUSH1)})
|
||||||
|
require.NoError(t, signTx(bc, notAddedTxes[i]))
|
||||||
|
require.NoError(t, bc.PoolTx(notAddedTxes[i]))
|
||||||
|
}
|
||||||
|
b := bc.newBlock(addedTxes...)
|
||||||
|
require.NoError(t, bc.AddBlock(b))
|
||||||
|
mempool := bc.GetMemPool()
|
||||||
|
for _, tx := range addedTxes {
|
||||||
|
require.False(t, mempool.ContainsKey(tx.Hash()))
|
||||||
|
}
|
||||||
|
for _, tx := range notAddedTxes {
|
||||||
|
require.True(t, mempool.ContainsKey(tx.Hash()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestHasBlock(t *testing.T) {
|
func TestHasBlock(t *testing.T) {
|
||||||
bc := newTestChain(t)
|
bc := newTestChain(t)
|
||||||
blocks, err := bc.genBlocks(50)
|
blocks, err := bc.genBlocks(50)
|
||||||
|
|
Loading…
Reference in a new issue