2020-01-15 07:52:59 +00:00
|
|
|
package mempool
|
2019-10-24 09:29:55 +00:00
|
|
|
|
|
|
|
import (
|
2020-07-09 09:57:24 +00:00
|
|
|
"math/big"
|
2020-02-06 14:20:38 +00:00
|
|
|
"sort"
|
2019-10-24 09:29:55 +00:00
|
|
|
"testing"
|
|
|
|
|
2020-06-18 09:00:51 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/config/netmode"
|
2020-03-03 14:21:42 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
2020-06-05 09:17:16 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/opcode"
|
2019-10-24 09:29:55 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
type FeerStub struct {
|
2020-06-23 14:15:35 +00:00
|
|
|
feePerByte int64
|
2019-10-24 09:29:55 +00:00
|
|
|
}
|
|
|
|
|
2020-07-09 09:57:24 +00:00
|
|
|
var balance = big.NewInt(10000000)
|
2020-06-23 14:15:35 +00:00
|
|
|
|
|
|
|
func (fs *FeerStub) FeePerByte() int64 {
|
2020-05-08 17:54:24 +00:00
|
|
|
return fs.feePerByte
|
2019-10-24 09:29:55 +00:00
|
|
|
}
|
|
|
|
|
2020-07-09 09:57:24 +00:00
|
|
|
func (fs *FeerStub) GetUtilityTokenBalance(uint160 util.Uint160) *big.Int {
|
2020-06-23 14:15:35 +00:00
|
|
|
return balance
|
2020-05-18 08:20:41 +00:00
|
|
|
}
|
|
|
|
|
2019-10-24 09:29:55 +00:00
|
|
|
func testMemPoolAddRemoveWithFeer(t *testing.T, fs Feer) {
|
|
|
|
mp := NewMemPool(10)
|
2020-06-18 09:00:51 +00:00
|
|
|
tx := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-04-22 17:42:38 +00:00
|
|
|
tx.Nonce = 0
|
2020-06-05 16:01:10 +00:00
|
|
|
_, ok := mp.TryGetValue(tx.Hash())
|
2019-10-24 09:29:55 +00:00
|
|
|
require.Equal(t, false, ok)
|
2020-02-05 11:24:36 +00:00
|
|
|
require.NoError(t, mp.Add(tx, fs))
|
2019-10-24 09:29:55 +00:00
|
|
|
// Re-adding should fail.
|
2020-02-05 11:24:36 +00:00
|
|
|
require.Error(t, mp.Add(tx, fs))
|
2020-06-05 16:01:10 +00:00
|
|
|
tx2, ok := mp.TryGetValue(tx.Hash())
|
2019-10-24 09:29:55 +00:00
|
|
|
require.Equal(t, true, ok)
|
|
|
|
require.Equal(t, tx, tx2)
|
|
|
|
mp.Remove(tx.Hash())
|
2020-06-05 16:01:10 +00:00
|
|
|
_, ok = mp.TryGetValue(tx.Hash())
|
2019-10-24 09:29:55 +00:00
|
|
|
require.Equal(t, false, ok)
|
|
|
|
// Make sure nothing left in the mempool after removal.
|
2020-02-05 14:13:35 +00:00
|
|
|
assert.Equal(t, 0, len(mp.verifiedMap))
|
|
|
|
assert.Equal(t, 0, len(mp.verifiedTxes))
|
2019-10-24 09:29:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestMemPoolAddRemove(t *testing.T) {
|
2020-06-18 19:32:29 +00:00
|
|
|
var fs = &FeerStub{}
|
|
|
|
testMemPoolAddRemoveWithFeer(t, fs)
|
2019-10-24 09:29:55 +00:00
|
|
|
}
|
2019-12-13 21:03:04 +00:00
|
|
|
|
2020-02-06 14:20:38 +00:00
|
|
|
func TestOverCapacity(t *testing.T) {
|
2020-06-18 19:32:29 +00:00
|
|
|
var fs = &FeerStub{}
|
2020-02-06 14:20:38 +00:00
|
|
|
const mempoolSize = 10
|
|
|
|
mp := NewMemPool(mempoolSize)
|
|
|
|
|
|
|
|
for i := 0; i < mempoolSize; i++ {
|
2020-06-18 09:00:51 +00:00
|
|
|
tx := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-04-22 17:42:38 +00:00
|
|
|
tx.Nonce = uint32(i)
|
2020-02-06 14:20:38 +00:00
|
|
|
require.NoError(t, mp.Add(tx, fs))
|
|
|
|
}
|
|
|
|
txcnt := uint32(mempoolSize)
|
|
|
|
require.Equal(t, mempoolSize, mp.Count())
|
|
|
|
require.Equal(t, true, sort.IsSorted(sort.Reverse(mp.verifiedTxes)))
|
|
|
|
|
|
|
|
// Fees are also prioritized.
|
2020-06-04 20:58:13 +00:00
|
|
|
for i := 0; i < mempoolSize; i++ {
|
2020-06-18 09:00:51 +00:00
|
|
|
tx := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-05-08 17:54:24 +00:00
|
|
|
tx.Attributes = append(tx.Attributes, transaction.Attribute{
|
|
|
|
Usage: transaction.Hash1,
|
|
|
|
Data: util.Uint256{1, 2, 3, 4}.BytesBE(),
|
|
|
|
})
|
2020-06-23 14:15:35 +00:00
|
|
|
tx.NetworkFee = 10000
|
2020-04-22 17:42:38 +00:00
|
|
|
tx.Nonce = txcnt
|
2020-02-06 14:20:38 +00:00
|
|
|
txcnt++
|
2020-06-23 14:15:35 +00:00
|
|
|
// size is 84, networkFee is 10000 => feePerByte is 119
|
2020-02-06 14:20:38 +00:00
|
|
|
require.NoError(t, mp.Add(tx, fs))
|
|
|
|
require.Equal(t, mempoolSize, mp.Count())
|
|
|
|
require.Equal(t, true, sort.IsSorted(sort.Reverse(mp.verifiedTxes)))
|
|
|
|
}
|
|
|
|
// Less prioritized txes are not allowed anymore.
|
2020-06-18 09:00:51 +00:00
|
|
|
tx := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-05-08 17:54:24 +00:00
|
|
|
tx.Attributes = append(tx.Attributes, transaction.Attribute{
|
|
|
|
Usage: transaction.Hash1,
|
|
|
|
Data: util.Uint256{1, 2, 3, 4}.BytesBE(),
|
|
|
|
})
|
2020-06-23 14:15:35 +00:00
|
|
|
tx.NetworkFee = 100
|
2020-04-22 17:42:38 +00:00
|
|
|
tx.Nonce = txcnt
|
2020-02-06 14:20:38 +00:00
|
|
|
txcnt++
|
|
|
|
require.Error(t, mp.Add(tx, fs))
|
|
|
|
require.Equal(t, mempoolSize, mp.Count())
|
|
|
|
require.Equal(t, true, sort.IsSorted(sort.Reverse(mp.verifiedTxes)))
|
|
|
|
|
|
|
|
// Low net fee, but higher per-byte fee is still a better combination.
|
2020-06-18 09:00:51 +00:00
|
|
|
tx = transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-04-22 17:42:38 +00:00
|
|
|
tx.Nonce = txcnt
|
2020-06-23 14:15:35 +00:00
|
|
|
tx.NetworkFee = 7000
|
2020-02-06 14:20:38 +00:00
|
|
|
txcnt++
|
2020-06-23 14:15:35 +00:00
|
|
|
// size is 51 (no attributes), networkFee is 7000 (<10000)
|
|
|
|
// => feePerByte is 137 (>119)
|
2020-02-06 14:20:38 +00:00
|
|
|
require.NoError(t, mp.Add(tx, fs))
|
|
|
|
require.Equal(t, mempoolSize, mp.Count())
|
|
|
|
require.Equal(t, true, sort.IsSorted(sort.Reverse(mp.verifiedTxes)))
|
|
|
|
|
|
|
|
// High priority always wins over low priority.
|
|
|
|
for i := 0; i < mempoolSize; i++ {
|
2020-06-18 09:00:51 +00:00
|
|
|
tx := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-06-23 14:15:35 +00:00
|
|
|
tx.NetworkFee = 8000
|
2020-04-22 17:42:38 +00:00
|
|
|
tx.Nonce = txcnt
|
2020-02-06 14:20:38 +00:00
|
|
|
txcnt++
|
|
|
|
require.NoError(t, mp.Add(tx, fs))
|
|
|
|
require.Equal(t, mempoolSize, mp.Count())
|
|
|
|
require.Equal(t, true, sort.IsSorted(sort.Reverse(mp.verifiedTxes)))
|
|
|
|
}
|
|
|
|
// Good luck with low priority now.
|
2020-06-18 09:00:51 +00:00
|
|
|
tx = transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-04-22 17:42:38 +00:00
|
|
|
tx.Nonce = txcnt
|
2020-06-23 14:15:35 +00:00
|
|
|
tx.NetworkFee = 7000
|
2020-02-06 14:20:38 +00:00
|
|
|
require.Error(t, mp.Add(tx, fs))
|
|
|
|
require.Equal(t, mempoolSize, mp.Count())
|
|
|
|
require.Equal(t, true, sort.IsSorted(sort.Reverse(mp.verifiedTxes)))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetVerified(t *testing.T) {
|
2020-06-18 19:32:29 +00:00
|
|
|
var fs = &FeerStub{}
|
2020-02-06 14:20:38 +00:00
|
|
|
const mempoolSize = 10
|
|
|
|
mp := NewMemPool(mempoolSize)
|
|
|
|
|
|
|
|
txes := make([]*transaction.Transaction, 0, mempoolSize)
|
|
|
|
for i := 0; i < mempoolSize; i++ {
|
2020-06-18 09:00:51 +00:00
|
|
|
tx := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-04-22 17:42:38 +00:00
|
|
|
tx.Nonce = uint32(i)
|
2020-02-06 14:20:38 +00:00
|
|
|
txes = append(txes, tx)
|
|
|
|
require.NoError(t, mp.Add(tx, fs))
|
|
|
|
}
|
|
|
|
require.Equal(t, mempoolSize, mp.Count())
|
|
|
|
verTxes := mp.GetVerifiedTransactions()
|
|
|
|
require.Equal(t, mempoolSize, len(verTxes))
|
2020-06-05 16:01:10 +00:00
|
|
|
require.ElementsMatch(t, txes, verTxes)
|
2020-02-06 14:20:38 +00:00
|
|
|
for _, tx := range txes {
|
|
|
|
mp.Remove(tx.Hash())
|
|
|
|
}
|
|
|
|
verTxes = mp.GetVerifiedTransactions()
|
|
|
|
require.Equal(t, 0, len(verTxes))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRemoveStale(t *testing.T) {
|
2020-06-18 19:32:29 +00:00
|
|
|
var fs = &FeerStub{}
|
2020-02-06 14:20:38 +00:00
|
|
|
const mempoolSize = 10
|
|
|
|
mp := NewMemPool(mempoolSize)
|
|
|
|
|
|
|
|
txes1 := make([]*transaction.Transaction, 0, mempoolSize/2)
|
|
|
|
txes2 := make([]*transaction.Transaction, 0, mempoolSize/2)
|
|
|
|
for i := 0; i < mempoolSize; i++ {
|
2020-06-18 09:00:51 +00:00
|
|
|
tx := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-04-22 17:42:38 +00:00
|
|
|
tx.Nonce = uint32(i)
|
2020-02-06 14:20:38 +00:00
|
|
|
if i%2 == 0 {
|
|
|
|
txes1 = append(txes1, tx)
|
|
|
|
} else {
|
|
|
|
txes2 = append(txes2, tx)
|
|
|
|
}
|
|
|
|
require.NoError(t, mp.Add(tx, fs))
|
|
|
|
}
|
|
|
|
require.Equal(t, mempoolSize, mp.Count())
|
|
|
|
mp.RemoveStale(func(t *transaction.Transaction) bool {
|
|
|
|
for _, tx := range txes2 {
|
|
|
|
if tx == t {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
2020-05-18 08:20:41 +00:00
|
|
|
}, &FeerStub{})
|
2020-02-06 14:20:38 +00:00
|
|
|
require.Equal(t, mempoolSize/2, mp.Count())
|
|
|
|
verTxes := mp.GetVerifiedTransactions()
|
2020-02-18 15:56:41 +00:00
|
|
|
for _, txf := range verTxes {
|
2020-06-05 16:01:10 +00:00
|
|
|
require.NotContains(t, txes1, txf)
|
|
|
|
require.Contains(t, txes2, txf)
|
2020-01-15 07:52:59 +00:00
|
|
|
}
|
|
|
|
}
|
2020-05-18 08:20:41 +00:00
|
|
|
|
|
|
|
func TestMemPoolFees(t *testing.T) {
|
|
|
|
mp := NewMemPool(10)
|
|
|
|
sender0 := util.Uint160{1, 2, 3}
|
2020-06-18 09:00:51 +00:00
|
|
|
tx0 := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-07-09 09:57:24 +00:00
|
|
|
tx0.NetworkFee = balance.Int64() + 1
|
2020-05-18 08:20:41 +00:00
|
|
|
tx0.Sender = sender0
|
|
|
|
// insufficient funds to add transaction, but balance should 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{
|
2020-06-23 14:15:35 +00:00
|
|
|
balance: balance,
|
2020-05-18 08:20:41 +00:00
|
|
|
feeSum: 0,
|
|
|
|
}, mp.fees[sender0])
|
|
|
|
|
2020-07-09 09:57:24 +00:00
|
|
|
balancePart := new(big.Int).Div(balance, big.NewInt(4))
|
2020-05-18 08:20:41 +00:00
|
|
|
// no problems with adding another transaction with lower fee
|
2020-06-18 09:00:51 +00:00
|
|
|
tx1 := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-07-09 09:57:24 +00:00
|
|
|
tx1.NetworkFee = balancePart.Int64()
|
2020-05-18 08:20:41 +00:00
|
|
|
tx1.Sender = sender0
|
|
|
|
require.NoError(t, mp.Add(tx1, &FeerStub{}))
|
|
|
|
require.Equal(t, 1, len(mp.fees))
|
|
|
|
require.Equal(t, utilityBalanceAndFees{
|
2020-06-23 14:15:35 +00:00
|
|
|
balance: balance,
|
|
|
|
feeSum: tx1.NetworkFee,
|
2020-05-18 08:20:41 +00:00
|
|
|
}, mp.fees[sender0])
|
|
|
|
|
|
|
|
// balance shouldn't change after adding one more transaction
|
2020-06-18 09:00:51 +00:00
|
|
|
tx2 := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-07-09 09:57:24 +00:00
|
|
|
tx2.NetworkFee = new(big.Int).Sub(balance, balancePart).Int64()
|
2020-05-18 08:20:41 +00:00
|
|
|
tx2.Sender = sender0
|
|
|
|
require.NoError(t, mp.Add(tx2, &FeerStub{}))
|
|
|
|
require.Equal(t, 2, len(mp.verifiedTxes))
|
|
|
|
require.Equal(t, 1, len(mp.fees))
|
|
|
|
require.Equal(t, utilityBalanceAndFees{
|
2020-06-23 14:15:35 +00:00
|
|
|
balance: balance,
|
2020-07-09 09:57:24 +00:00
|
|
|
feeSum: balance.Int64(),
|
2020-05-18 08:20:41 +00:00
|
|
|
}, mp.fees[sender0])
|
|
|
|
|
|
|
|
// can't add more transactions as we don't have enough GAS
|
2020-06-18 09:00:51 +00:00
|
|
|
tx3 := transaction.New(netmode.UnitTestNet, []byte{byte(opcode.PUSH1)}, 0)
|
2020-06-23 14:15:35 +00:00
|
|
|
tx3.NetworkFee = 1
|
2020-05-18 08:20:41 +00:00
|
|
|
tx3.Sender = sender0
|
|
|
|
require.Equal(t, false, mp.Verify(tx3, &FeerStub{}))
|
|
|
|
require.Error(t, mp.Add(tx3, &FeerStub{}))
|
|
|
|
require.Equal(t, 1, len(mp.fees))
|
|
|
|
require.Equal(t, utilityBalanceAndFees{
|
2020-06-23 14:15:35 +00:00
|
|
|
balance: balance,
|
2020-07-09 09:57:24 +00:00
|
|
|
feeSum: balance.Int64(),
|
2020-05-18 08:20:41 +00:00
|
|
|
}, mp.fees[sender0])
|
|
|
|
|
|
|
|
// check whether sender's fee updates correctly
|
|
|
|
mp.RemoveStale(func(t *transaction.Transaction) bool {
|
|
|
|
if t == tx2 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}, &FeerStub{})
|
|
|
|
require.Equal(t, 1, len(mp.fees))
|
|
|
|
require.Equal(t, utilityBalanceAndFees{
|
2020-06-23 14:15:35 +00:00
|
|
|
balance: balance,
|
|
|
|
feeSum: tx2.NetworkFee,
|
2020-05-18 08:20:41 +00:00
|
|
|
}, mp.fees[sender0])
|
|
|
|
|
|
|
|
// there should be nothing left
|
|
|
|
mp.RemoveStale(func(t *transaction.Transaction) bool {
|
|
|
|
if t == tx3 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}, &FeerStub{})
|
|
|
|
require.Equal(t, 0, len(mp.fees))
|
|
|
|
}
|