mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-06 02:52:19 +00:00
eec5fa4d41
* Add eth/67 * Listen to eth/66 on a separate port * Fix compilation error * Fix cfg66.ListenAddr * Update erigon ports in README * Expose port 30304 in docker * P2pProtocolVersionFlag instead of second sentry * Remove "66 by default" from usage * Small comment
294 lines
7.9 KiB
Go
294 lines
7.9 KiB
Go
package sentry
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"math"
|
|
"math/big"
|
|
"strings"
|
|
"syscall"
|
|
|
|
proto_sentry "github.com/ledgerwatch/erigon-lib/gointerfaces/sentry"
|
|
types2 "github.com/ledgerwatch/erigon-lib/gointerfaces/types"
|
|
"github.com/ledgerwatch/erigon/common"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/eth/protocols/eth"
|
|
"github.com/ledgerwatch/erigon/p2p"
|
|
"github.com/ledgerwatch/erigon/rlp"
|
|
"github.com/ledgerwatch/erigon/turbo/stages/headerdownload"
|
|
"github.com/ledgerwatch/log/v3"
|
|
"google.golang.org/grpc"
|
|
)
|
|
|
|
// Methods of sentry called by Core
|
|
|
|
const (
|
|
// This is the target size for the packs of transactions or announcements. A
|
|
// pack can get larger than this if a single transactions exceeds this size.
|
|
maxTxPacketSize = 100 * 1024
|
|
)
|
|
|
|
func (cs *MultiClient) PropagateNewBlockHashes(ctx context.Context, announces []headerdownload.Announce) {
|
|
cs.lock.RLock()
|
|
defer cs.lock.RUnlock()
|
|
typedRequest := make(eth.NewBlockHashesPacket, len(announces))
|
|
for i := range announces {
|
|
typedRequest[i].Hash = announces[i].Hash
|
|
typedRequest[i].Number = announces[i].Number
|
|
}
|
|
data, err := rlp.EncodeToBytes(&typedRequest)
|
|
if err != nil {
|
|
log.Error("propagateNewBlockHashes", "err", err)
|
|
return
|
|
}
|
|
var req66 *proto_sentry.OutboundMessageData
|
|
// Send the block to a subset of our peers
|
|
sendToAmount := int(math.Sqrt(float64(len(cs.sentries))))
|
|
for i, sentry := range cs.sentries {
|
|
if !sentry.Ready() {
|
|
continue
|
|
}
|
|
if i > sendToAmount { //TODO: send to random sentries, not just to fi
|
|
break
|
|
}
|
|
|
|
switch sentry.Protocol() {
|
|
|
|
case eth.ETH66, eth.ETH67:
|
|
if req66 == nil {
|
|
req66 = &proto_sentry.OutboundMessageData{
|
|
Id: proto_sentry.MessageId_NEW_BLOCK_HASHES_66,
|
|
Data: data,
|
|
}
|
|
|
|
_, err = sentry.SendMessageToAll(ctx, req66, &grpc.EmptyCallOption{})
|
|
if err != nil {
|
|
log.Error("propagateNewBlockHashes", "err", err)
|
|
}
|
|
}
|
|
default:
|
|
//??
|
|
}
|
|
}
|
|
}
|
|
|
|
func (cs *MultiClient) BroadcastNewBlock(ctx context.Context, block *types.Block, td *big.Int) {
|
|
cs.lock.RLock()
|
|
defer cs.lock.RUnlock()
|
|
data, err := rlp.EncodeToBytes(ð.NewBlockPacket{
|
|
Block: block,
|
|
TD: td,
|
|
})
|
|
if err != nil {
|
|
log.Error("broadcastNewBlock", "err", err)
|
|
}
|
|
var req66 *proto_sentry.SendMessageToRandomPeersRequest
|
|
// Send the block to a subset of our peers
|
|
sendToAmount := int(math.Sqrt(float64(len(cs.sentries))))
|
|
for i, sentry := range cs.sentries {
|
|
if !sentry.Ready() {
|
|
continue
|
|
}
|
|
if i > sendToAmount { //TODO: send to random sentries, not just to fi
|
|
break
|
|
}
|
|
|
|
switch sentry.Protocol() {
|
|
|
|
case eth.ETH66, eth.ETH67:
|
|
if req66 == nil {
|
|
req66 = &proto_sentry.SendMessageToRandomPeersRequest{
|
|
MaxPeers: 1024,
|
|
Data: &proto_sentry.OutboundMessageData{
|
|
Id: proto_sentry.MessageId_NEW_BLOCK_66,
|
|
Data: data,
|
|
},
|
|
}
|
|
}
|
|
if _, err = sentry.SendMessageToRandomPeers(ctx, req66, &grpc.EmptyCallOption{}); err != nil {
|
|
if isPeerNotFoundErr(err) || networkTemporaryErr(err) {
|
|
log.Debug("broadcastNewBlock", "err", err)
|
|
continue
|
|
}
|
|
log.Error("broadcastNewBlock", "err", err)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (cs *MultiClient) BroadcastLocalPooledTxs(ctx context.Context, txs []common.Hash) {
|
|
if len(txs) == 0 {
|
|
return
|
|
}
|
|
|
|
cs.lock.RLock()
|
|
defer cs.lock.RUnlock()
|
|
initialAmount := len(txs)
|
|
avgPeersPerSent65 := 0
|
|
avgPeersPerSent66 := 0
|
|
initialTxs := txs
|
|
for len(txs) > 0 {
|
|
|
|
pendingLen := maxTxPacketSize / common.HashLength
|
|
pending := make([]common.Hash, 0, pendingLen)
|
|
|
|
for i := 0; i < pendingLen && i < len(txs); i++ {
|
|
pending = append(pending, txs[i])
|
|
}
|
|
txs = txs[len(pending):]
|
|
|
|
data, err := rlp.EncodeToBytes(eth.NewPooledTransactionHashesPacket(pending))
|
|
if err != nil {
|
|
log.Error("BroadcastLocalPooledTxs", "err", err)
|
|
}
|
|
var req66 *proto_sentry.OutboundMessageData
|
|
// Send the block to a subset of our peers
|
|
sendToAmount := int(math.Sqrt(float64(len(cs.sentries))))
|
|
for i, sentry := range cs.sentries {
|
|
if !sentry.Ready() {
|
|
continue
|
|
}
|
|
if i > sendToAmount { //TODO: send to random sentries, not just to fi
|
|
break
|
|
}
|
|
|
|
switch sentry.Protocol() {
|
|
case eth.ETH66, eth.ETH67:
|
|
if req66 == nil {
|
|
req66 = &proto_sentry.OutboundMessageData{
|
|
Id: proto_sentry.MessageId_NEW_POOLED_TRANSACTION_HASHES_66,
|
|
Data: data,
|
|
}
|
|
}
|
|
peers, err := sentry.SendMessageToAll(ctx, req66, &grpc.EmptyCallOption{})
|
|
if err != nil {
|
|
if isPeerNotFoundErr(err) || networkTemporaryErr(err) {
|
|
log.Debug("BroadcastLocalPooledTxs", "err", err)
|
|
continue
|
|
}
|
|
log.Error("BroadcastLocalPooledTxs", "err", err)
|
|
}
|
|
avgPeersPerSent66 += len(peers.GetPeers())
|
|
}
|
|
}
|
|
}
|
|
if initialAmount == 1 {
|
|
log.Info("local tx propagated", "to_peers_amount", avgPeersPerSent65+avgPeersPerSent66, "tx_hash", initialTxs[0].String())
|
|
} else {
|
|
log.Info("local txs propagated", "to_peers_amount", avgPeersPerSent65+avgPeersPerSent66, "txs_amount", initialAmount)
|
|
}
|
|
}
|
|
|
|
func (cs *MultiClient) BroadcastRemotePooledTxs(ctx context.Context, txs []common.Hash) {
|
|
if len(txs) == 0 {
|
|
return
|
|
}
|
|
|
|
cs.lock.RLock()
|
|
defer cs.lock.RUnlock()
|
|
for len(txs) > 0 {
|
|
|
|
pendingLen := maxTxPacketSize / common.HashLength
|
|
pending := make([]common.Hash, 0, pendingLen)
|
|
|
|
for i := 0; i < pendingLen && i < len(txs); i++ {
|
|
pending = append(pending, txs[i])
|
|
}
|
|
txs = txs[len(pending):]
|
|
|
|
data, err := rlp.EncodeToBytes(eth.NewPooledTransactionHashesPacket(pending))
|
|
if err != nil {
|
|
log.Error("BroadcastRemotePooledTxs", "err", err)
|
|
}
|
|
var req66 *proto_sentry.SendMessageToRandomPeersRequest
|
|
// Send the block to a subset of our peers
|
|
sendToAmount := int(math.Sqrt(float64(len(cs.sentries))))
|
|
for i, sentry := range cs.sentries {
|
|
if !sentry.Ready() {
|
|
continue
|
|
}
|
|
if i > sendToAmount { //TODO: send to random sentries, not just to fi
|
|
break
|
|
}
|
|
|
|
switch sentry.Protocol() {
|
|
|
|
case eth.ETH66, eth.ETH67:
|
|
if req66 == nil {
|
|
req66 = &proto_sentry.SendMessageToRandomPeersRequest{
|
|
MaxPeers: 1024,
|
|
Data: &proto_sentry.OutboundMessageData{
|
|
Id: proto_sentry.MessageId_NEW_POOLED_TRANSACTION_HASHES_66,
|
|
Data: data,
|
|
},
|
|
}
|
|
}
|
|
if _, err = sentry.SendMessageToRandomPeers(ctx, req66, &grpc.EmptyCallOption{}); err != nil {
|
|
if isPeerNotFoundErr(err) || networkTemporaryErr(err) {
|
|
log.Debug("BroadcastRemotePooledTxs", "err", err)
|
|
continue
|
|
}
|
|
log.Error("BroadcastRemotePooledTxs", "err", err)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (cs *MultiClient) PropagatePooledTxsToPeersList(ctx context.Context, peers []*types2.H512, txs []common.Hash) {
|
|
if len(txs) == 0 {
|
|
return
|
|
}
|
|
|
|
cs.lock.RLock()
|
|
defer cs.lock.RUnlock()
|
|
for len(txs) > 0 {
|
|
|
|
pendingLen := maxTxPacketSize / common.HashLength
|
|
pending := make([]common.Hash, 0, pendingLen)
|
|
|
|
for i := 0; i < pendingLen && i < len(txs); i++ {
|
|
pending = append(pending, txs[i])
|
|
}
|
|
txs = txs[len(pending):]
|
|
|
|
data, err := rlp.EncodeToBytes(eth.NewPooledTransactionHashesPacket(pending))
|
|
if err != nil {
|
|
log.Error("PropagatePooledTxsToPeersList", "err", err)
|
|
}
|
|
for _, sentry := range cs.sentries {
|
|
if !sentry.Ready() {
|
|
continue
|
|
}
|
|
|
|
for _, peer := range peers {
|
|
switch sentry.Protocol() {
|
|
|
|
case eth.ETH66, eth.ETH67:
|
|
req66 := &proto_sentry.SendMessageByIdRequest{
|
|
PeerId: peer,
|
|
Data: &proto_sentry.OutboundMessageData{
|
|
Id: proto_sentry.MessageId_NEW_POOLED_TRANSACTION_HASHES_66,
|
|
Data: data,
|
|
},
|
|
}
|
|
if _, err = sentry.SendMessageById(ctx, req66, &grpc.EmptyCallOption{}); err != nil {
|
|
if isPeerNotFoundErr(err) || networkTemporaryErr(err) {
|
|
log.Debug("PropagatePooledTxsToPeersList", "err", err)
|
|
continue
|
|
}
|
|
log.Error("PropagatePooledTxsToPeersList", "err", err)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func networkTemporaryErr(err error) bool {
|
|
return errors.Is(err, syscall.EPIPE) || errors.Is(err, p2p.ErrShuttingDown)
|
|
}
|
|
func isPeerNotFoundErr(err error) bool {
|
|
return strings.Contains(err.Error(), "peer not found")
|
|
}
|