Skip to content

Commit d6900b1

Browse files
authored
Update ReapMaxBytesMaxGas to include estimated gas (#263)
* checkpoint: add gasEstimate to WrappedTx in mempool * updated proto with gas estimated + build blocks with gas used + mempool tests * fix replay stub * implement fallback to gasWanted if gasEstimate is not set * remove print * fix some tests * fix some tests * fix nondeterministic test * remove both maxGasWanted and maxGasEstimated, just use maxGas * fix test: mempool mocks * fix race: pending tx mempool size bytes
1 parent e18f602 commit d6900b1

File tree

8 files changed

+425
-282
lines changed

8 files changed

+425
-282
lines changed

abci/types/types.pb.go

+290-254
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

internal/mempool/mempool.go

+15-3
Original file line numberDiff line numberDiff line change
@@ -198,7 +198,7 @@ func (txmp *TxMempool) BytesNotPending() int64 {
198198
}
199199

200200
func (txmp *TxMempool) TotalTxsBytesSize() int64 {
201-
return txmp.BytesNotPending() + int64(txmp.pendingTxs.sizeBytes)
201+
return txmp.BytesNotPending() + int64(txmp.pendingTxs.SizeBytes())
202202
}
203203

204204
// PendingSize returns the number of pending transactions in the mempool.
@@ -335,6 +335,7 @@ func (txmp *TxMempool) CheckTx(
335335
evmAddress: res.EVMSenderAddress,
336336
isEVM: res.IsEVM,
337337
removeHandler: removeHandler,
338+
estimatedGas: res.GasEstimated,
338339
}
339340

340341
if err == nil {
@@ -452,9 +453,19 @@ func (txmp *TxMempool) ReapMaxBytesMaxGas(maxBytes, maxGas, minTxsInBlock int64)
452453
if maxBytes > -1 && totalSize+size > maxBytes {
453454
return false
454455
}
456+
457+
// if the tx doesn't have a gas estimate, fallback to gas wanted
458+
var gasEstimateOrWanted int64
459+
if wtx.estimatedGas > 0 {
460+
gasEstimateOrWanted = wtx.estimatedGas
461+
} else {
462+
gasEstimateOrWanted = wtx.gasWanted
463+
}
464+
455465
totalSize += size
456-
gas := totalGas + wtx.gasWanted
457-
if nonzeroGasTxCnt >= minTxsInBlock && maxGas > -1 && gas > maxGas {
466+
gas := totalGas + gasEstimateOrWanted
467+
maxGasExceeded := maxGas > -1 && gas > maxGas
468+
if nonzeroGasTxCnt >= minTxsInBlock && maxGasExceeded {
458469
return false
459470
}
460471

@@ -679,6 +690,7 @@ func (txmp *TxMempool) addNewTransaction(wtx *WrappedTx, res *abci.ResponseCheck
679690
}
680691

681692
wtx.gasWanted = res.GasWanted
693+
wtx.estimatedGas = res.GasEstimated
682694
wtx.priority = priority
683695
wtx.sender = sender
684696
wtx.peers = map[uint16]struct{}{

internal/mempool/mempool_test.go

+103-20
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,8 @@ import (
3030
type application struct {
3131
*kvstore.Application
3232

33+
gasWanted *int64
34+
gasEstimated *int64
3335
occupiedNonces map[string][]uint64
3436
}
3537

@@ -38,13 +40,26 @@ type testTx struct {
3840
priority int64
3941
}
4042

43+
var DefaultGasEstimated = int64(1)
44+
var DefaultGasWanted = int64(1)
45+
4146
func (app *application) CheckTx(_ context.Context, req *abci.RequestCheckTx) (*abci.ResponseCheckTxV2, error) {
4247

4348
var (
4449
priority int64
4550
sender string
4651
)
4752

53+
gasWanted := DefaultGasWanted
54+
if app.gasWanted != nil {
55+
gasWanted = *app.gasWanted
56+
}
57+
58+
gasEstimated := DefaultGasEstimated
59+
if app.gasEstimated != nil {
60+
gasEstimated = *app.gasEstimated
61+
}
62+
4863
if strings.HasPrefix(string(req.Tx), "evm") {
4964
// format is evm-sender-0=account=priority=nonce
5065
// split into respective vars
@@ -55,18 +70,20 @@ func (app *application) CheckTx(_ context.Context, req *abci.RequestCheckTx) (*a
5570
if err != nil {
5671
// could not parse
5772
return &abci.ResponseCheckTxV2{ResponseCheckTx: &abci.ResponseCheckTx{
58-
Priority: priority,
59-
Code: 100,
60-
GasWanted: 1,
73+
Priority: priority,
74+
Code: 100,
75+
GasWanted: gasWanted,
76+
GasEstimated: gasEstimated,
6177
}}, nil
6278
}
6379
nonce, err := strconv.ParseUint(string(parts[3]), 10, 64)
6480
if err != nil {
6581
// could not parse
6682
return &abci.ResponseCheckTxV2{ResponseCheckTx: &abci.ResponseCheckTx{
67-
Priority: priority,
68-
Code: 101,
69-
GasWanted: 1,
83+
Priority: priority,
84+
Code: 101,
85+
GasWanted: gasWanted,
86+
GasEstimated: gasEstimated,
7087
}}, nil
7188
}
7289
if app.occupiedNonces == nil {
@@ -92,9 +109,10 @@ func (app *application) CheckTx(_ context.Context, req *abci.RequestCheckTx) (*a
92109
app.occupiedNonces[account] = append(app.occupiedNonces[account], nonce)
93110
return &abci.ResponseCheckTxV2{
94111
ResponseCheckTx: &abci.ResponseCheckTx{
95-
Priority: v,
96-
Code: code.CodeTypeOK,
97-
GasWanted: 1,
112+
Priority: v,
113+
Code: code.CodeTypeOK,
114+
GasWanted: gasWanted,
115+
GasEstimated: gasEstimated,
98116
},
99117
EVMNonce: nonce,
100118
EVMSenderAddress: account,
@@ -122,26 +140,29 @@ func (app *application) CheckTx(_ context.Context, req *abci.RequestCheckTx) (*a
122140
v, err := strconv.ParseInt(string(parts[2]), 10, 64)
123141
if err != nil {
124142
return &abci.ResponseCheckTxV2{ResponseCheckTx: &abci.ResponseCheckTx{
125-
Priority: priority,
126-
Code: 100,
127-
GasWanted: 1,
143+
Priority: priority,
144+
Code: 100,
145+
GasWanted: gasWanted,
146+
GasEstimated: gasEstimated,
128147
}}, nil
129148
}
130149

131150
priority = v
132151
sender = string(parts[0])
133152
} else {
134153
return &abci.ResponseCheckTxV2{ResponseCheckTx: &abci.ResponseCheckTx{
135-
Priority: priority,
136-
Code: 101,
137-
GasWanted: 1,
154+
Priority: priority,
155+
Code: 101,
156+
GasWanted: gasWanted,
157+
GasEstimated: gasEstimated,
138158
}}, nil
139159
}
140160
return &abci.ResponseCheckTxV2{ResponseCheckTx: &abci.ResponseCheckTx{
141-
Priority: priority,
142-
Sender: sender,
143-
Code: code.CodeTypeOK,
144-
GasWanted: 1,
161+
Priority: priority,
162+
Sender: sender,
163+
Code: code.CodeTypeOK,
164+
GasWanted: gasWanted,
165+
GasEstimated: gasEstimated,
145166
}}, nil
146167
}
147168

@@ -346,7 +367,9 @@ func TestTxMempool_ReapMaxBytesMaxGas(t *testing.T) {
346367
ctx, cancel := context.WithCancel(context.Background())
347368
defer cancel()
348369

349-
client := abciclient.NewLocalClient(log.NewNopLogger(), &application{Application: kvstore.NewApplication()})
370+
// gas estimated is 1 so we will use this and not fallback to gas wanted
371+
gasEstimated := int64(1)
372+
client := abciclient.NewLocalClient(log.NewNopLogger(), &application{Application: kvstore.NewApplication(), gasEstimated: &gasEstimated})
350373
if err := client.Start(ctx); err != nil {
351374
t.Fatal(err)
352375
}
@@ -424,6 +447,66 @@ func TestTxMempool_ReapMaxBytesMaxGas(t *testing.T) {
424447
require.Len(t, reapedTxs, 2)
425448
}()
426449

450+
// Reap by max gas estimated
451+
wg.Add(1)
452+
go func() {
453+
defer wg.Done()
454+
reapedTxs := txmp.ReapMaxBytesMaxGas(-1, 50, 0)
455+
ensurePrioritized(reapedTxs)
456+
require.Equal(t, len(tTxs), txmp.Size())
457+
require.Len(t, reapedTxs, 50)
458+
}()
459+
460+
wg.Wait()
461+
}
462+
463+
func TestTxMempool_ReapMaxBytesMaxGas_FallbackToGasWanted(t *testing.T) {
464+
ctx, cancel := context.WithCancel(context.Background())
465+
defer cancel()
466+
467+
// gas estimated is 0 so we will fallback to gas wanted
468+
gasEstimated := int64(0)
469+
client := abciclient.NewLocalClient(log.NewNopLogger(), &application{Application: kvstore.NewApplication(), gasEstimated: &gasEstimated})
470+
if err := client.Start(ctx); err != nil {
471+
t.Fatal(err)
472+
}
473+
t.Cleanup(client.Wait)
474+
475+
txmp := setup(t, client, 0)
476+
tTxs := checkTxs(ctx, t, txmp, 100, 0)
477+
478+
txMap := make(map[types.TxKey]testTx)
479+
priorities := make([]int64, len(tTxs))
480+
for i, tTx := range tTxs {
481+
txMap[tTx.tx.Key()] = tTx
482+
priorities[i] = tTx.priority
483+
}
484+
485+
// Debug: Print sorted priorities
486+
sort.Slice(priorities, func(i, j int) bool {
487+
return priorities[i] > priorities[j]
488+
})
489+
490+
ensurePrioritized := func(reapedTxs types.Txs) {
491+
reapedPriorities := make([]int64, len(reapedTxs))
492+
for i, rTx := range reapedTxs {
493+
reapedPriorities[i] = txMap[rTx.Key()].priority
494+
}
495+
496+
require.Equal(t, priorities[:len(reapedPriorities)], reapedPriorities)
497+
}
498+
499+
var wg sync.WaitGroup
500+
501+
wg.Add(1)
502+
go func() {
503+
defer wg.Done()
504+
reapedTxs := txmp.ReapMaxBytesMaxGas(-1, 50, 0)
505+
ensurePrioritized(reapedTxs)
506+
require.Equal(t, len(tTxs), txmp.Size())
507+
require.Len(t, reapedTxs, 50)
508+
}()
509+
427510
wg.Wait()
428511
}
429512

internal/mempool/mocks/mempool.go

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

internal/mempool/tx.go

+9
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,9 @@ type WrappedTx struct {
3939
// gasWanted defines the amount of gas the transaction sender requires
4040
gasWanted int64
4141

42+
// estimatedGas defines the amount of gas that the transaction is estimated to use
43+
estimatedGas int64
44+
4245
// priority defines the transaction's priority as specified by the application
4346
// in the ResponseCheckTx response.
4447
priority int64
@@ -397,6 +400,12 @@ func (p *PendingTxs) Insert(tx *WrappedTx, resCheckTx *abci.ResponseCheckTxV2, t
397400
return nil
398401
}
399402

403+
func (p *PendingTxs) SizeBytes() uint64 {
404+
p.mtx.RLock()
405+
defer p.mtx.RUnlock()
406+
return p.sizeBytes
407+
}
408+
400409
func (p *PendingTxs) Peek(max int) []TxWithResponse {
401410
p.mtx.RLock()
402411
defer p.mtx.RUnlock()

internal/mempool/types.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -44,9 +44,9 @@ type Mempool interface {
4444

4545
// ReapMaxBytesMaxGas reaps transactions from the mempool up to maxBytes
4646
// bytes total with the condition that the total gasWanted must be less than
47-
// maxGas.
47+
// maxGas and that the total estimated gas used is less than maxGasEstimated.
4848
//
49-
// If both maxes are negative, there is no cap on the size of all returned
49+
// If all 3 maxes are negative, there is no cap on the size of all returned
5050
// transactions (~ all available transactions).
5151
ReapMaxBytesMaxGas(maxBytes, maxGas, minTxsInBlock int64) types.Txs
5252

internal/p2p/peermanager_test.go

+4-2
Original file line numberDiff line numberDiff line change
@@ -994,8 +994,10 @@ func TestPeerManager_Dialed_MaxConnectedUpgrade(t *testing.T) {
994994
// Start upgrade with c
995995
dial, err := peerManager.TryDialNext()
996996
require.NoError(t, err)
997-
require.Equal(t, c, dial)
998-
require.NoError(t, peerManager.Dialed(c))
997+
if dial != c && dial != d {
998+
t.Fatalf("dial = %s, expected %s or %s", dial, c, d)
999+
}
1000+
require.NoError(t, peerManager.Dialed(dial))
9991001

10001002
// Try to dial d - should fail since we're at upgrade capacity
10011003
dial, err = peerManager.TryDialNext()

proto/tendermint/abci/types.proto

+1
Original file line numberDiff line numberDiff line change
@@ -297,6 +297,7 @@ message ResponseCheckTx {
297297
string codespace = 8;
298298
string sender = 9;
299299
int64 priority = 10;
300+
int64 gas_estimated = 12;
300301

301302
reserved 4, 6, 7, 11; // see https://github.com/tendermint/tendermint/issues/8543
302303
}

0 commit comments

Comments
 (0)