mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-08 12:01:20 +00:00
142 lines
4.1 KiB
Go
142 lines
4.1 KiB
Go
|
package commands
|
|||
|
|
|||
|
import (
|
|||
|
"bytes"
|
|||
|
"context"
|
|||
|
"fmt"
|
|||
|
|
|||
|
"github.com/ledgerwatch/erigon/common"
|
|||
|
"github.com/ledgerwatch/erigon/core/types"
|
|||
|
"github.com/ledgerwatch/erigon/ethdb"
|
|||
|
proto_txpool "github.com/ledgerwatch/erigon/gointerfaces/txpool"
|
|||
|
"github.com/ledgerwatch/erigon/rlp"
|
|||
|
)
|
|||
|
|
|||
|
// NetAPI the interface for the net_ RPC commands
|
|||
|
type TxPoolAPI interface {
|
|||
|
Content(ctx context.Context) (map[string]map[string]map[string]*RPCTransaction, error)
|
|||
|
}
|
|||
|
|
|||
|
// TxPoolAPIImpl data structure to store things needed for net_ commands
|
|||
|
type TxPoolAPIImpl struct {
|
|||
|
*BaseAPI
|
|||
|
pool proto_txpool.TxpoolClient
|
|||
|
db ethdb.RoKV
|
|||
|
}
|
|||
|
|
|||
|
// NewTxPoolAPI returns NetAPIImplImpl instance
|
|||
|
func NewTxPoolAPI(base *BaseAPI, db ethdb.RoKV, pool proto_txpool.TxpoolClient) *TxPoolAPIImpl {
|
|||
|
return &TxPoolAPIImpl{
|
|||
|
BaseAPI: base,
|
|||
|
pool: pool,
|
|||
|
db: db,
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
func (api *TxPoolAPIImpl) Content(ctx context.Context) (map[string]map[string]map[string]*RPCTransaction, error) {
|
|||
|
reply, err := api.pool.All(ctx, &proto_txpool.AllRequest{})
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
|
|||
|
content := map[string]map[string]map[string]*RPCTransaction{
|
|||
|
"pending": make(map[string]map[string]*RPCTransaction),
|
|||
|
"queued": make(map[string]map[string]*RPCTransaction),
|
|||
|
}
|
|||
|
|
|||
|
pending := make(map[common.Address][]types.Transaction, 8)
|
|||
|
queued := make(map[common.Address][]types.Transaction, 8)
|
|||
|
for i := range reply.Txs {
|
|||
|
stream := rlp.NewStream(bytes.NewReader(reply.Txs[i].RlpTx), 0)
|
|||
|
txn, err := types.DecodeTransaction(stream)
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
addr := common.BytesToAddress(reply.Txs[i].Sender)
|
|||
|
switch reply.Txs[i].Type {
|
|||
|
case proto_txpool.AllReply_PENDING:
|
|||
|
if _, ok := pending[addr]; !ok {
|
|||
|
pending[addr] = make([]types.Transaction, 0, 4)
|
|||
|
}
|
|||
|
pending[addr] = append(pending[addr], txn)
|
|||
|
case proto_txpool.AllReply_QUEUED:
|
|||
|
if _, ok := queued[addr]; !ok {
|
|||
|
queued[addr] = make([]types.Transaction, 0, 4)
|
|||
|
}
|
|||
|
queued[addr] = append(queued[addr], txn)
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
tx, err := api.db.BeginRo(ctx)
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
defer tx.Rollback()
|
|||
|
|
|||
|
//curHeader := rawdb.ReadCurrentHeader(tx)
|
|||
|
// Flatten the pending transactions
|
|||
|
for account, txs := range pending {
|
|||
|
dump := make(map[string]*RPCTransaction)
|
|||
|
for _, txn := range txs {
|
|||
|
dump[fmt.Sprintf("%d", txn.GetNonce())] = newRPCPendingTransaction(txn)
|
|||
|
}
|
|||
|
content["pending"][account.Hex()] = dump
|
|||
|
}
|
|||
|
// Flatten the queued transactions
|
|||
|
for account, txs := range queued {
|
|||
|
dump := make(map[string]*RPCTransaction)
|
|||
|
for _, txn := range txs {
|
|||
|
dump[fmt.Sprintf("%d", txn.GetNonce())] = newRPCPendingTransaction(txn)
|
|||
|
}
|
|||
|
content["queued"][account.Hex()] = dump
|
|||
|
}
|
|||
|
return content, nil
|
|||
|
}
|
|||
|
|
|||
|
/*
|
|||
|
|
|||
|
// Status returns the number of pending and queued transaction in the pool.
|
|||
|
func (s *PublicTxPoolAPI) Status() map[string]hexutil.Uint {
|
|||
|
pending, queue := s.b.Stats()
|
|||
|
return map[string]hexutil.Uint{
|
|||
|
"pending": hexutil.Uint(pending),
|
|||
|
"queued": hexutil.Uint(queue),
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Inspect retrieves the content of the transaction pool and flattens it into an
|
|||
|
// easily inspectable list.
|
|||
|
func (s *PublicTxPoolAPI) Inspect() map[string]map[string]map[string]string {
|
|||
|
content := map[string]map[string]map[string]string{
|
|||
|
"pending": make(map[string]map[string]string),
|
|||
|
"queued": make(map[string]map[string]string),
|
|||
|
}
|
|||
|
pending, queue := s.b.TxPoolContent()
|
|||
|
|
|||
|
// Define a formatter to flatten a transaction into a string
|
|||
|
var format = func(tx *types.Transaction) string {
|
|||
|
if to := tx.To(); to != nil {
|
|||
|
return fmt.Sprintf("%s: %v wei + %v gas × %v wei", tx.To().Hex(), tx.Value(), tx.Gas(), tx.GasPrice())
|
|||
|
}
|
|||
|
return fmt.Sprintf("contract creation: %v wei + %v gas × %v wei", tx.Value(), tx.Gas(), tx.GasPrice())
|
|||
|
}
|
|||
|
// Flatten the pending transactions
|
|||
|
for account, txs := range pending {
|
|||
|
dump := make(map[string]string)
|
|||
|
for _, tx := range txs {
|
|||
|
dump[fmt.Sprintf("%d", tx.Nonce())] = format(tx)
|
|||
|
}
|
|||
|
content["pending"][account.Hex()] = dump
|
|||
|
}
|
|||
|
// Flatten the queued transactions
|
|||
|
for account, txs := range queue {
|
|||
|
dump := make(map[string]string)
|
|||
|
for _, tx := range txs {
|
|||
|
dump[fmt.Sprintf("%d", tx.Nonce())] = format(tx)
|
|||
|
}
|
|||
|
content["queued"][account.Hex()] = dump
|
|||
|
}
|
|||
|
return content
|
|||
|
}
|
|||
|
*/
|