mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-14 23:08:20 +00:00
431 lines
14 KiB
Go
431 lines
14 KiB
Go
package txpool
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"math/rand"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/direct"
|
|
"github.com/ledgerwatch/erigon-lib/gointerfaces"
|
|
proto_sentry "github.com/ledgerwatch/erigon-lib/gointerfaces/sentry"
|
|
"github.com/ledgerwatch/erigon/common"
|
|
"github.com/ledgerwatch/erigon/common/debug"
|
|
"github.com/ledgerwatch/erigon/core"
|
|
"github.com/ledgerwatch/erigon/eth/fetcher"
|
|
"github.com/ledgerwatch/erigon/eth/protocols/eth"
|
|
"github.com/ledgerwatch/erigon/rlp"
|
|
"github.com/ledgerwatch/erigon/turbo/stages/txpropagate"
|
|
"github.com/ledgerwatch/log/v3"
|
|
"google.golang.org/grpc"
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/status"
|
|
"google.golang.org/protobuf/types/known/emptypb"
|
|
)
|
|
|
|
// P2PServer - receiving and sending messages to Sentries
|
|
type P2PServer struct {
|
|
ctx context.Context
|
|
Sentries []direct.SentryClient
|
|
TxPool *core.TxPool
|
|
TxFetcher *fetcher.TxFetcher
|
|
RecentPeers *txpropagate.RecentlyConnectedPeers
|
|
}
|
|
|
|
func NewP2PServer(ctx context.Context, sentries []direct.SentryClient, txPool *core.TxPool) (*P2PServer, error) {
|
|
cs := &P2PServer{
|
|
ctx: ctx,
|
|
Sentries: sentries,
|
|
TxPool: txPool,
|
|
RecentPeers: &txpropagate.RecentlyConnectedPeers{},
|
|
}
|
|
|
|
return cs, nil
|
|
}
|
|
|
|
func (tp *P2PServer) newPooledTransactionHashes66(ctx context.Context, inreq *proto_sentry.InboundMessage, sentry direct.SentryClient) error {
|
|
var query eth.NewPooledTransactionHashesPacket
|
|
if err := rlp.DecodeBytes(inreq.Data, &query); err != nil {
|
|
return fmt.Errorf("decoding newPooledTransactionHashes66: %w, data: %x", err, inreq.Data)
|
|
}
|
|
return tp.TxFetcher.Notify(string(gointerfaces.ConvertH512ToBytes(inreq.PeerId)), query)
|
|
}
|
|
|
|
func (tp *P2PServer) newPooledTransactionHashes65(ctx context.Context, inreq *proto_sentry.InboundMessage, sentry direct.SentryClient) error {
|
|
var query eth.NewPooledTransactionHashesPacket
|
|
if err := rlp.DecodeBytes(inreq.Data, &query); err != nil {
|
|
return fmt.Errorf("decoding newPooledTransactionHashes65: %w, data: %x", err, inreq.Data)
|
|
}
|
|
return tp.TxFetcher.Notify(string(gointerfaces.ConvertH512ToBytes(inreq.PeerId)), query)
|
|
}
|
|
|
|
func (tp *P2PServer) pooledTransactions66(ctx context.Context, inreq *proto_sentry.InboundMessage, sentry direct.SentryClient) error {
|
|
txs := ð.PooledTransactionsPacket66{}
|
|
if err := txs.DecodeRLP(rlp.NewStream(bytes.NewReader(inreq.Data), 0)); err != nil {
|
|
return fmt.Errorf("decoding pooledTransactions66: %w, data: %x", err, inreq.Data)
|
|
}
|
|
|
|
return tp.TxFetcher.Enqueue(string(gointerfaces.ConvertH512ToBytes(inreq.PeerId)), txs.PooledTransactionsPacket, true)
|
|
}
|
|
|
|
func (tp *P2PServer) pooledTransactions65(ctx context.Context, inreq *proto_sentry.InboundMessage, sentry direct.SentryClient) error {
|
|
txs := ð.PooledTransactionsPacket{}
|
|
if err := txs.DecodeRLP(rlp.NewStream(bytes.NewReader(inreq.Data), 0)); err != nil {
|
|
return fmt.Errorf("decoding pooledTransactions65: %w, data: %x", err, inreq.Data)
|
|
}
|
|
|
|
return tp.TxFetcher.Enqueue(string(gointerfaces.ConvertH512ToBytes(inreq.PeerId)), *txs, true)
|
|
}
|
|
|
|
func (tp *P2PServer) transactions66(ctx context.Context, inreq *proto_sentry.InboundMessage, sentry direct.SentryClient) error {
|
|
return tp.transactions65(ctx, inreq, sentry)
|
|
}
|
|
|
|
func (tp *P2PServer) transactions65(ctx context.Context, inreq *proto_sentry.InboundMessage, sentry direct.SentryClient) error {
|
|
if tp.TxPool == nil {
|
|
return nil
|
|
}
|
|
var query eth.TransactionsPacket
|
|
if err := rlp.DecodeBytes(inreq.Data, &query); err != nil {
|
|
return fmt.Errorf("decoding TransactionsPacket: %w, data: %x", err, inreq.Data)
|
|
}
|
|
return tp.TxFetcher.Enqueue(string(gointerfaces.ConvertH512ToBytes(inreq.PeerId)), query, false)
|
|
}
|
|
|
|
func (tp *P2PServer) getPooledTransactions66(ctx context.Context, inreq *proto_sentry.InboundMessage, sentry direct.SentryClient) error {
|
|
if tp.TxPool == nil {
|
|
return nil
|
|
}
|
|
var query eth.GetPooledTransactionsPacket66
|
|
if err := rlp.DecodeBytes(inreq.Data, &query); err != nil {
|
|
return fmt.Errorf("decoding GetPooledTransactionsPacket66: %w, data: %x", err, inreq.Data)
|
|
}
|
|
_, txs := eth.AnswerGetPooledTransactions(tp.TxPool, query.GetPooledTransactionsPacket)
|
|
b, err := rlp.EncodeToBytes(ð.PooledTransactionsRLPPacket66{
|
|
RequestId: query.RequestId,
|
|
PooledTransactionsRLPPacket: txs,
|
|
})
|
|
if err != nil {
|
|
return fmt.Errorf("encode GetPooledTransactionsPacket66 response: %w", err)
|
|
}
|
|
// TODO: implement logic from perr.ReplyPooledTransactionsRLP - to remember tx ids
|
|
outreq := proto_sentry.SendMessageByIdRequest{
|
|
PeerId: inreq.PeerId,
|
|
Data: &proto_sentry.OutboundMessageData{Id: proto_sentry.MessageId_POOLED_TRANSACTIONS_66, Data: b},
|
|
}
|
|
_, err = sentry.SendMessageById(ctx, &outreq, &grpc.EmptyCallOption{})
|
|
if err != nil {
|
|
return fmt.Errorf("send pooled transactions response: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (tp *P2PServer) getPooledTransactions65(ctx context.Context, inreq *proto_sentry.InboundMessage, sentry direct.SentryClient) error {
|
|
if tp.TxPool == nil {
|
|
return nil
|
|
}
|
|
var query eth.GetPooledTransactionsPacket
|
|
if err := rlp.DecodeBytes(inreq.Data, &query); err != nil {
|
|
return fmt.Errorf("decoding getPooledTransactions65: %w, data: %x", err, inreq.Data)
|
|
}
|
|
_, txs := eth.AnswerGetPooledTransactions(tp.TxPool, query)
|
|
b, err := rlp.EncodeToBytes(eth.PooledTransactionsRLPPacket(txs))
|
|
if err != nil {
|
|
return fmt.Errorf("encode getPooledTransactions65 response: %w", err)
|
|
}
|
|
// TODO: implement logic from perr.ReplyPooledTransactionsRLP - to remember tx ids
|
|
outreq := proto_sentry.SendMessageByIdRequest{
|
|
PeerId: inreq.PeerId,
|
|
Data: &proto_sentry.OutboundMessageData{Id: proto_sentry.MessageId_POOLED_TRANSACTIONS_65, Data: b},
|
|
}
|
|
_, err = sentry.SendMessageById(ctx, &outreq, &grpc.EmptyCallOption{})
|
|
if err != nil {
|
|
return fmt.Errorf("send pooled transactions response: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (tp *P2PServer) SendTxsRequest(ctx context.Context, peerID string, hashes []common.Hash) []byte {
|
|
var outreq65, outreq66 *proto_sentry.SendMessageByIdRequest
|
|
|
|
// if sentry not found peers to send such message, try next one. stop if found.
|
|
for i, ok, next := tp.randSentryIndex(); ok; i, ok = next() {
|
|
if !tp.Sentries[i].Ready() {
|
|
continue
|
|
}
|
|
|
|
switch tp.Sentries[i].Protocol() {
|
|
case eth.ETH65:
|
|
if outreq65 == nil {
|
|
data65, err := rlp.EncodeToBytes(eth.GetPooledTransactionsPacket(hashes))
|
|
if err != nil {
|
|
log.Error("Could not encode transactions request", "err", err)
|
|
return nil
|
|
}
|
|
|
|
outreq65 = &proto_sentry.SendMessageByIdRequest{
|
|
PeerId: gointerfaces.ConvertBytesToH512([]byte(peerID)),
|
|
Data: &proto_sentry.OutboundMessageData{Id: proto_sentry.MessageId_GET_POOLED_TRANSACTIONS_65, Data: data65},
|
|
}
|
|
}
|
|
|
|
if sentPeers, err1 := tp.Sentries[i].SendMessageById(ctx, outreq65, &grpc.EmptyCallOption{}); err1 != nil {
|
|
if isPeerNotFoundErr(err1) {
|
|
continue
|
|
}
|
|
log.Error("[SendTxsRequest]", "err", err1)
|
|
} else if sentPeers != nil && len(sentPeers.Peers) != 0 {
|
|
return gointerfaces.ConvertH512ToBytes(sentPeers.Peers[0])
|
|
}
|
|
case eth.ETH66:
|
|
if outreq66 == nil {
|
|
data66, err := rlp.EncodeToBytes(ð.GetPooledTransactionsPacket66{
|
|
RequestId: rand.Uint64(), //nolint:gosec
|
|
GetPooledTransactionsPacket: hashes,
|
|
})
|
|
if err != nil {
|
|
log.Error("Could not encode transactions request", "err", err)
|
|
return nil
|
|
}
|
|
|
|
outreq66 = &proto_sentry.SendMessageByIdRequest{
|
|
PeerId: gointerfaces.ConvertBytesToH512([]byte(peerID)),
|
|
Data: &proto_sentry.OutboundMessageData{Id: proto_sentry.MessageId_GET_POOLED_TRANSACTIONS_66, Data: data66},
|
|
}
|
|
}
|
|
|
|
if sentPeers, err1 := tp.Sentries[i].SendMessageById(ctx, outreq66, &grpc.EmptyCallOption{}); err1 != nil {
|
|
if isPeerNotFoundErr(err1) {
|
|
continue
|
|
}
|
|
log.Error("[SendTxsRequest]", "err", err1)
|
|
|
|
} else if sentPeers != nil && len(sentPeers.Peers) != 0 {
|
|
return gointerfaces.ConvertH512ToBytes(sentPeers.Peers[0])
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (tp *P2PServer) randSentryIndex() (int, bool, func() (int, bool)) {
|
|
var i int
|
|
if len(tp.Sentries) > 1 {
|
|
i = rand.Intn(len(tp.Sentries))
|
|
}
|
|
to := i
|
|
return i, true, func() (int, bool) {
|
|
i = (i + 1) % len(tp.Sentries)
|
|
return i, i != to
|
|
}
|
|
}
|
|
|
|
func (tp *P2PServer) HandleInboundMessage(ctx context.Context, inreq *proto_sentry.InboundMessage, sentry direct.SentryClient) error {
|
|
switch inreq.Id {
|
|
|
|
// ==== eth 65 ====
|
|
case proto_sentry.MessageId_TRANSACTIONS_65:
|
|
return tp.transactions65(ctx, inreq, sentry)
|
|
case proto_sentry.MessageId_NEW_POOLED_TRANSACTION_HASHES_65:
|
|
return tp.newPooledTransactionHashes65(ctx, inreq, sentry)
|
|
case proto_sentry.MessageId_GET_POOLED_TRANSACTIONS_65:
|
|
return tp.getPooledTransactions65(ctx, inreq, sentry)
|
|
case proto_sentry.MessageId_POOLED_TRANSACTIONS_65:
|
|
return tp.pooledTransactions65(ctx, inreq, sentry)
|
|
|
|
// ==== eth 66 ====
|
|
case proto_sentry.MessageId_NEW_POOLED_TRANSACTION_HASHES_66:
|
|
return tp.newPooledTransactionHashes66(ctx, inreq, sentry)
|
|
case proto_sentry.MessageId_POOLED_TRANSACTIONS_66:
|
|
return tp.pooledTransactions66(ctx, inreq, sentry)
|
|
case proto_sentry.MessageId_TRANSACTIONS_66:
|
|
return tp.transactions66(ctx, inreq, sentry)
|
|
case proto_sentry.MessageId_GET_POOLED_TRANSACTIONS_66:
|
|
return tp.getPooledTransactions66(ctx, inreq, sentry)
|
|
default:
|
|
return fmt.Errorf("not implemented for message Id: %s", inreq.Id)
|
|
}
|
|
}
|
|
|
|
func RecvTxMessageLoop(ctx context.Context, sentry direct.SentryClient, handleInboundMessage func(ctx context.Context, inreq *proto_sentry.InboundMessage, sentry direct.SentryClient) error, wg *sync.WaitGroup) {
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
default:
|
|
}
|
|
|
|
if _, err := sentry.HandShake(ctx, &emptypb.Empty{}, grpc.WaitForReady(true)); err != nil {
|
|
s, ok := status.FromError(err)
|
|
doLog := !((ok && s.Code() == codes.Canceled) || errors.Is(err, io.EOF) || errors.Is(err, context.Canceled))
|
|
if doLog {
|
|
log.Error("[RecvTxMessage] sentry not ready yet", "err", err)
|
|
}
|
|
time.Sleep(time.Second)
|
|
continue
|
|
}
|
|
if err := RecvTxMessage(ctx, sentry, handleInboundMessage, wg); err != nil {
|
|
s, ok := status.FromError(err)
|
|
doLog := !((ok && s.Code() == codes.Canceled) || errors.Is(err, io.EOF) || errors.Is(err, context.Canceled))
|
|
if doLog {
|
|
log.Error("[RecvTxMessage]", "err", err)
|
|
}
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
|
|
// RecvTxMessage
|
|
// wg is used only in tests to avoid time.Sleep. For non-test code wg == nil
|
|
func RecvTxMessage(ctx context.Context,
|
|
sentry direct.SentryClient,
|
|
handleInboundMessage func(ctx context.Context, inreq *proto_sentry.InboundMessage, sentry direct.SentryClient) error,
|
|
wg *sync.WaitGroup,
|
|
) (err error) {
|
|
defer func() { err = debug.ReportPanicAndRecover(err) }() // avoid crash because Erigon's core does many things
|
|
streamCtx, cancel := context.WithCancel(ctx)
|
|
defer cancel()
|
|
|
|
stream, err := sentry.Messages(streamCtx, &proto_sentry.MessagesRequest{Ids: []proto_sentry.MessageId{
|
|
eth.ToProto[eth.ETH65][eth.NewPooledTransactionHashesMsg],
|
|
eth.ToProto[eth.ETH65][eth.GetPooledTransactionsMsg],
|
|
eth.ToProto[eth.ETH65][eth.TransactionsMsg],
|
|
eth.ToProto[eth.ETH65][eth.PooledTransactionsMsg],
|
|
|
|
eth.ToProto[eth.ETH66][eth.NewPooledTransactionHashesMsg],
|
|
eth.ToProto[eth.ETH66][eth.GetPooledTransactionsMsg],
|
|
eth.ToProto[eth.ETH66][eth.TransactionsMsg],
|
|
eth.ToProto[eth.ETH66][eth.PooledTransactionsMsg],
|
|
}}, grpc.WaitForReady(true))
|
|
if err != nil {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
default:
|
|
}
|
|
return err
|
|
}
|
|
|
|
var req *proto_sentry.InboundMessage
|
|
for req, err = stream.Recv(); ; req, err = stream.Recv() {
|
|
if err != nil {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
default:
|
|
}
|
|
return err
|
|
}
|
|
if req == nil {
|
|
return
|
|
}
|
|
if err = handleInboundMessage(ctx, req, sentry); err != nil {
|
|
s, ok := status.FromError(err)
|
|
doLog := !((ok && s.Code() == codes.Canceled) || errors.Is(err, io.EOF) || errors.Is(err, context.Canceled))
|
|
if doLog {
|
|
if rlp.IsDecodeError(err) {
|
|
log.Debug("[RecvTxMessage] Handling incoming message", "error", err)
|
|
} else {
|
|
log.Warn("[RecvTxMessage] Handling incoming message", "error", err)
|
|
}
|
|
}
|
|
}
|
|
if wg != nil {
|
|
wg.Done()
|
|
}
|
|
}
|
|
}
|
|
|
|
func RecvPeersLoop(ctx context.Context, sentry direct.SentryClient, recentPeers *txpropagate.RecentlyConnectedPeers, wg *sync.WaitGroup) {
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
default:
|
|
}
|
|
|
|
if _, err := sentry.HandShake(ctx, &emptypb.Empty{}, grpc.WaitForReady(true)); err != nil {
|
|
s, ok := status.FromError(err)
|
|
doLog := !((ok && s.Code() == codes.Canceled) || errors.Is(err, io.EOF) || errors.Is(err, context.Canceled))
|
|
if doLog {
|
|
log.Warn("[RecvPeers] sentry not ready yet", "err", err)
|
|
}
|
|
time.Sleep(time.Second)
|
|
continue
|
|
}
|
|
if err := RecvPeers(ctx, sentry, recentPeers, wg); err != nil {
|
|
s, ok := status.FromError(err)
|
|
doLog := !((ok && s.Code() == codes.Canceled) || errors.Is(err, io.EOF) || errors.Is(err, context.Canceled))
|
|
if doLog {
|
|
log.Warn("[RecvPeers]", "err", err)
|
|
}
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
|
|
// RecvPeers
|
|
// wg is used only in tests to avoid time.Sleep. For non-test code wg == nil
|
|
func RecvPeers(ctx context.Context,
|
|
sentry direct.SentryClient,
|
|
recentPeers *txpropagate.RecentlyConnectedPeers,
|
|
wg *sync.WaitGroup,
|
|
) (err error) {
|
|
defer func() { err = debug.ReportPanicAndRecover(err) }() // avoid crash because Erigon's core does many things
|
|
streamCtx, cancel := context.WithCancel(ctx)
|
|
defer cancel()
|
|
|
|
stream, err := sentry.Peers(streamCtx, &proto_sentry.PeersRequest{})
|
|
if err != nil {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
default:
|
|
}
|
|
if s, ok := status.FromError(err); ok && s.Code() == codes.Canceled {
|
|
return
|
|
}
|
|
if errors.Is(err, io.EOF) || errors.Is(err, context.Canceled) {
|
|
return
|
|
}
|
|
return err
|
|
}
|
|
|
|
var req *proto_sentry.PeersReply
|
|
for req, err = stream.Recv(); ; req, err = stream.Recv() {
|
|
if err != nil {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
default:
|
|
}
|
|
if s, ok := status.FromError(err); ok && s.Code() == codes.Canceled {
|
|
return
|
|
}
|
|
if errors.Is(err, io.EOF) || errors.Is(err, context.Canceled) {
|
|
return
|
|
}
|
|
return err
|
|
}
|
|
if req == nil {
|
|
return
|
|
}
|
|
switch req.Event {
|
|
case proto_sentry.PeersReply_Connect:
|
|
recentPeers.AddPeer(req.PeerId)
|
|
}
|
|
if wg != nil {
|
|
wg.Done()
|
|
}
|
|
}
|
|
}
|
|
|
|
func isPeerNotFoundErr(err error) bool {
|
|
return strings.Contains(err.Error(), "peer not found")
|
|
}
|