mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-19 00:54:12 +00:00
b160e30652
* empty engine * start sync from peer right after handshake if it's TD is higher * fixCanonicalChain to print logs * clean * >
226 lines
6.8 KiB
Go
226 lines
6.8 KiB
Go
package download
|
|
|
|
import (
|
|
"context"
|
|
"math/rand"
|
|
|
|
"github.com/holiman/uint256"
|
|
"github.com/ledgerwatch/erigon/common"
|
|
"github.com/ledgerwatch/erigon/eth/protocols/eth"
|
|
"github.com/ledgerwatch/erigon/gointerfaces"
|
|
proto_sentry "github.com/ledgerwatch/erigon/gointerfaces/sentry"
|
|
"github.com/ledgerwatch/erigon/log"
|
|
"github.com/ledgerwatch/erigon/rlp"
|
|
"github.com/ledgerwatch/erigon/turbo/stages/bodydownload"
|
|
"github.com/ledgerwatch/erigon/turbo/stages/headerdownload"
|
|
"google.golang.org/grpc"
|
|
)
|
|
|
|
// Methods of sentry called by Core
|
|
|
|
func (cs *ControlServerImpl) UpdateHead(ctx context.Context, height uint64, hash common.Hash, td *uint256.Int) {
|
|
cs.lock.Lock()
|
|
defer cs.lock.Unlock()
|
|
cs.headHeight = height
|
|
cs.headHash = hash
|
|
cs.headTd = td
|
|
statusMsg := makeStatusData(cs)
|
|
for _, sentry := range cs.sentries {
|
|
if !sentry.Ready() {
|
|
continue
|
|
}
|
|
|
|
if _, err := sentry.SetStatus(ctx, statusMsg, &grpc.EmptyCallOption{}); err != nil {
|
|
log.Error("Update status message for the sentry", "error", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (cs *ControlServerImpl) SendBodyRequest(ctx context.Context, req *bodydownload.BodyRequest) []byte {
|
|
// if sentry not found peers to send such message, try next one. stop if found.
|
|
for i, ok, next := cs.randSentryIndex(); ok; i, ok = next() {
|
|
if !cs.sentries[i].Ready() {
|
|
continue
|
|
}
|
|
|
|
switch cs.sentries[i].Protocol() {
|
|
case eth.ETH66:
|
|
//log.Info(fmt.Sprintf("Sending body request for %v", req.BlockNums))
|
|
var bytes []byte
|
|
var err error
|
|
bytes, err = rlp.EncodeToBytes(ð.GetBlockBodiesPacket66{
|
|
RequestId: rand.Uint64(),
|
|
GetBlockBodiesPacket: req.Hashes,
|
|
})
|
|
if err != nil {
|
|
log.Error("Could not encode block bodies request", "err", err)
|
|
return nil
|
|
}
|
|
outreq := proto_sentry.SendMessageByMinBlockRequest{
|
|
MinBlock: req.BlockNums[len(req.BlockNums)-1],
|
|
Data: &proto_sentry.OutboundMessageData{
|
|
Id: proto_sentry.MessageId_GET_BLOCK_BODIES_66,
|
|
Data: bytes,
|
|
},
|
|
}
|
|
|
|
sentPeers, err1 := cs.sentries[i].SendMessageByMinBlock(ctx, &outreq, &grpc.EmptyCallOption{})
|
|
if err1 != nil {
|
|
log.Error("Could not send block bodies request", "err", err1)
|
|
return nil
|
|
}
|
|
if sentPeers == nil || len(sentPeers.Peers) == 0 {
|
|
continue
|
|
}
|
|
return gointerfaces.ConvertH512ToBytes(sentPeers.Peers[0])
|
|
case eth.ETH65:
|
|
//log.Info(fmt.Sprintf("Sending body request for %v", req.BlockNums))
|
|
var bytes []byte
|
|
var err error
|
|
bytes, err = rlp.EncodeToBytes(eth.GetBlockBodiesPacket(req.Hashes))
|
|
if err != nil {
|
|
log.Error("Could not encode block bodies request", "err", err)
|
|
return nil
|
|
}
|
|
outreq := proto_sentry.SendMessageByMinBlockRequest{
|
|
MinBlock: req.BlockNums[len(req.BlockNums)-1],
|
|
Data: &proto_sentry.OutboundMessageData{
|
|
Id: proto_sentry.MessageId_GET_BLOCK_BODIES_65,
|
|
Data: bytes,
|
|
},
|
|
}
|
|
|
|
sentPeers, err1 := cs.sentries[i].SendMessageByMinBlock(ctx, &outreq, &grpc.EmptyCallOption{})
|
|
if err1 != nil {
|
|
log.Error("Could not send block bodies request", "err", err1)
|
|
return nil
|
|
}
|
|
if sentPeers == nil || len(sentPeers.Peers) == 0 {
|
|
continue
|
|
}
|
|
return gointerfaces.ConvertH512ToBytes(sentPeers.Peers[0])
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (cs *ControlServerImpl) SendHeaderRequest(ctx context.Context, req *headerdownload.HeaderRequest) []byte {
|
|
// if sentry not found peers to send such message, try next one. stop if found.
|
|
for i, ok, next := cs.randSentryIndex(); ok; i, ok = next() {
|
|
if !cs.sentries[i].Ready() {
|
|
continue
|
|
}
|
|
switch cs.sentries[i].Protocol() {
|
|
case eth.ETH66:
|
|
//log.Info(fmt.Sprintf("Sending header request {hash: %x, height: %d, length: %d}", req.Hash, req.Number, req.Length))
|
|
reqData := ð.GetBlockHeadersPacket66{
|
|
RequestId: rand.Uint64(),
|
|
GetBlockHeadersPacket: ð.GetBlockHeadersPacket{
|
|
Amount: req.Length,
|
|
Reverse: req.Reverse,
|
|
Skip: req.Skip,
|
|
Origin: eth.HashOrNumber{Hash: req.Hash},
|
|
},
|
|
}
|
|
if req.Hash == (common.Hash{}) {
|
|
reqData.Origin.Number = req.Number
|
|
}
|
|
bytes, err := rlp.EncodeToBytes(reqData)
|
|
if err != nil {
|
|
log.Error("Could not encode header request", "err", err)
|
|
return nil
|
|
}
|
|
minBlock := req.Number
|
|
if !req.Reverse {
|
|
minBlock = req.Number + req.Length*req.Skip
|
|
}
|
|
|
|
outreq := proto_sentry.SendMessageByMinBlockRequest{
|
|
MinBlock: minBlock,
|
|
Data: &proto_sentry.OutboundMessageData{
|
|
Id: proto_sentry.MessageId_GET_BLOCK_HEADERS_66,
|
|
Data: bytes,
|
|
},
|
|
}
|
|
sentPeers, err1 := cs.sentries[i].SendMessageByMinBlock(ctx, &outreq, &grpc.EmptyCallOption{})
|
|
if err1 != nil {
|
|
log.Error("Could not send header request", "err", err1)
|
|
return nil
|
|
}
|
|
if sentPeers == nil || len(sentPeers.Peers) == 0 {
|
|
continue
|
|
}
|
|
return gointerfaces.ConvertH512ToBytes(sentPeers.Peers[0])
|
|
case eth.ETH65:
|
|
//log.Info(fmt.Sprintf("Sending header request {hash: %x, height: %d, length: %d}", req.Hash, req.Number, req.Length))
|
|
reqData := ð.GetBlockHeadersPacket{
|
|
Amount: req.Length,
|
|
Reverse: req.Reverse,
|
|
Skip: req.Skip,
|
|
Origin: eth.HashOrNumber{Hash: req.Hash},
|
|
}
|
|
if req.Hash == (common.Hash{}) {
|
|
reqData.Origin.Number = req.Number
|
|
}
|
|
bytes, err := rlp.EncodeToBytes(reqData)
|
|
if err != nil {
|
|
log.Error("Could not encode header request", "err", err)
|
|
return nil
|
|
}
|
|
minBlock := req.Number
|
|
if !req.Reverse {
|
|
minBlock = req.Number + req.Length*req.Skip
|
|
}
|
|
|
|
outreq := proto_sentry.SendMessageByMinBlockRequest{
|
|
MinBlock: minBlock,
|
|
Data: &proto_sentry.OutboundMessageData{
|
|
Id: proto_sentry.MessageId_GET_BLOCK_HEADERS_65,
|
|
Data: bytes,
|
|
},
|
|
}
|
|
sentPeers, err1 := cs.sentries[i].SendMessageByMinBlock(ctx, &outreq, &grpc.EmptyCallOption{})
|
|
if err1 != nil {
|
|
log.Error("Could not send header request", "err", err1)
|
|
return nil
|
|
}
|
|
if sentPeers == nil || len(sentPeers.Peers) == 0 {
|
|
continue
|
|
}
|
|
return gointerfaces.ConvertH512ToBytes(sentPeers.Peers[0])
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (cs *ControlServerImpl) randSentryIndex() (int, bool, func() (int, bool)) {
|
|
var i int
|
|
if len(cs.sentries) > 1 {
|
|
i = rand.Intn(len(cs.sentries) - 1)
|
|
}
|
|
to := i
|
|
return i, true, func() (int, bool) {
|
|
i = (i + 1) % len(cs.sentries)
|
|
return i, i != to
|
|
}
|
|
}
|
|
|
|
// sending list of penalties to all sentries
|
|
func (cs *ControlServerImpl) Penalize(ctx context.Context, penalties []headerdownload.PenaltyItem) {
|
|
for i := range penalties {
|
|
outreq := proto_sentry.PenalizePeerRequest{
|
|
PeerId: gointerfaces.ConvertBytesToH512([]byte(penalties[i].PeerID)),
|
|
Penalty: proto_sentry.PenaltyKind_Kick, // TODO: Extend penalty kinds
|
|
}
|
|
for i, ok, next := cs.randSentryIndex(); ok; i, ok = next() {
|
|
if !cs.sentries[i].Ready() {
|
|
continue
|
|
}
|
|
|
|
if _, err1 := cs.sentries[i].PenalizePeer(ctx, &outreq, &grpc.EmptyCallOption{}); err1 != nil {
|
|
log.Error("Could not send penalty", "err", err1)
|
|
}
|
|
}
|
|
}
|
|
}
|