erigon-pulse/turbo/txpool/p2p.go
2021-10-04 22:16:52 +07:00

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 := &eth.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 := &eth.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(&eth.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(&eth.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")
}