erigon-pulse/cl/phase1/network/gossip_manager.go
a 436493350e
Sentinel refactor (#8296)
1. changes sentinel to use an http-like interface

2. moves hexutil, crypto/blake2b, metrics packages to erigon-lib
2023-10-22 01:17:18 +02:00

188 lines
6.1 KiB
Go

package network
import (
"context"
"fmt"
"github.com/ledgerwatch/erigon-lib/common"
"sync"
"github.com/ledgerwatch/erigon/cl/freezer"
"github.com/ledgerwatch/erigon/cl/phase1/forkchoice"
"github.com/ledgerwatch/erigon/cl/sentinel/peers"
"github.com/ledgerwatch/erigon-lib/gointerfaces/sentinel"
"github.com/ledgerwatch/erigon-lib/types/ssz"
"github.com/ledgerwatch/erigon/cl/clparams"
"github.com/ledgerwatch/erigon/cl/cltypes"
"github.com/ledgerwatch/erigon/cl/utils"
"github.com/ledgerwatch/log/v3"
)
// Gossip manager is sending all messages to fork choice or others
type GossipManager struct {
recorder freezer.Freezer
forkChoice *forkchoice.ForkChoiceStore
sentinel sentinel.SentinelClient
// configs
beaconConfig *clparams.BeaconChainConfig
genesisConfig *clparams.GenesisConfig
mu sync.RWMutex
subs map[int]chan *peers.PeeredObject[*cltypes.SignedBeaconBlock]
totalSubs int
}
func NewGossipReceiver(s sentinel.SentinelClient, forkChoice *forkchoice.ForkChoiceStore,
beaconConfig *clparams.BeaconChainConfig, genesisConfig *clparams.GenesisConfig, recorder freezer.Freezer) *GossipManager {
return &GossipManager{
sentinel: s,
forkChoice: forkChoice,
beaconConfig: beaconConfig,
genesisConfig: genesisConfig,
recorder: recorder,
subs: make(map[int]chan *peers.PeeredObject[*cltypes.SignedBeaconBlock]),
}
}
// this subscribes to signed beacon blocks..... i wish this was better
func (g *GossipManager) SubscribeSignedBeaconBlocks(ctx context.Context) <-chan *peers.PeeredObject[*cltypes.SignedBeaconBlock] {
// a really big limit because why not....
out := make(chan *peers.PeeredObject[*cltypes.SignedBeaconBlock], 512)
g.mu.Lock()
g.totalSubs++
idx := g.totalSubs
g.subs[idx] = out
g.mu.Unlock()
go func() {
<-ctx.Done()
g.mu.Lock()
delete(g.subs, idx)
g.mu.Unlock()
}()
return out
}
func operationsContract[T ssz.EncodableSSZ](ctx context.Context, g *GossipManager, l log.Ctx, data *sentinel.GossipData, version int, name string, fn func(T, bool) error) error {
var t T
object := t.Clone().(T)
if err := object.DecodeSSZ(common.CopyBytes(data.Data), version); err != nil {
g.sentinel.BanPeer(ctx, data.Peer)
l["at"] = fmt.Sprintf("decoding %s", name)
return err
}
if err := fn(object /*test=*/, false); err != nil {
l["at"] = fmt.Sprintf("verify %s", name)
return err
}
if _, err := g.sentinel.PublishGossip(ctx, data); err != nil {
log.Debug("failed publish gossip", "err", err)
}
return nil
}
func (g *GossipManager) onRecv(ctx context.Context, data *sentinel.GossipData, l log.Ctx) (err error) {
defer func() {
r := recover()
if r != nil {
err = fmt.Errorf("%v", r)
}
}()
currentEpoch := utils.GetCurrentEpoch(g.genesisConfig.GenesisTime, g.beaconConfig.SecondsPerSlot, g.beaconConfig.SlotsPerEpoch)
version := g.beaconConfig.GetCurrentStateVersion(currentEpoch)
// Depending on the type of the received data, we create an instance of a specific type that implements the ObjectSSZ interface,
// then attempts to deserialize the received data into it.
// If the deserialization fails, an error is logged and the loop returns to the next iteration.
// If the deserialization is successful, the object is set to the deserialized value and the loop returns to the next iteration.
var object ssz.Unmarshaler
switch data.Type {
case sentinel.GossipType_BeaconBlockGossipType:
object = cltypes.NewSignedBeaconBlock(g.beaconConfig)
if err := object.DecodeSSZ(common.CopyBytes(data.Data), int(version)); err != nil {
g.sentinel.BanPeer(ctx, data.Peer)
l["at"] = "decoding block"
return err
}
block := object.(*cltypes.SignedBeaconBlock)
l["slot"] = block.Block.Slot
currentSlotByTime := utils.GetCurrentSlot(g.genesisConfig.GenesisTime, g.beaconConfig.SecondsPerSlot)
maxGossipSlotThreshold := uint64(4)
// Skip if slot is too far behind.
if block.Block.Slot+maxGossipSlotThreshold < currentSlotByTime {
return nil
}
if block.Block.Slot == currentSlotByTime {
if _, err := g.sentinel.PublishGossip(ctx, data); err != nil {
log.Debug("failed publish gossip", "err", err)
}
}
count, err := g.sentinel.GetPeers(ctx, &sentinel.EmptyMessage{})
if err != nil {
l["at"] = "sentinel peer count"
return err
}
log.Debug("Received block via gossip",
"peers", count.Amount,
"slot", block.Block.Slot,
)
if err := freezer.PutObjectSSZIntoFreezer("signedBeaconBlock", "caplin_core", block.Block.Slot, block, g.recorder); err != nil {
return err
}
g.mu.RLock()
for _, v := range g.subs {
select {
case v <- &peers.PeeredObject[*cltypes.SignedBeaconBlock]{Data: block, Peer: data.Peer.Pid}:
default:
}
}
g.mu.RUnlock()
case sentinel.GossipType_VoluntaryExitGossipType:
if err := operationsContract[*cltypes.SignedVoluntaryExit](ctx, g, l, data, int(version), "voluntary exit", g.forkChoice.OnVoluntaryExit); err != nil {
return err
}
case sentinel.GossipType_ProposerSlashingGossipType:
if err := operationsContract[*cltypes.ProposerSlashing](ctx, g, l, data, int(version), "proposer slashing", g.forkChoice.OnProposerSlashing); err != nil {
return err
}
case sentinel.GossipType_AttesterSlashingGossipType:
if err := operationsContract[*cltypes.AttesterSlashing](ctx, g, l, data, int(version), "attester slashing", g.forkChoice.OnAttesterSlashing); err != nil {
return err
}
case sentinel.GossipType_BlsToExecutionChangeGossipType:
if err := operationsContract[*cltypes.SignedBLSToExecutionChange](ctx, g, l, data, int(version), "bls to execution change", g.forkChoice.OnBlsToExecutionChange); err != nil {
return err
}
}
return nil
}
func (g *GossipManager) Start(ctx context.Context) {
subscription, err := g.sentinel.SubscribeGossip(ctx, &sentinel.EmptyMessage{})
if err != nil {
return
}
l := log.Ctx{}
for {
data, err := subscription.Recv()
if err != nil {
log.Warn("[Beacon Gossip] Fatal error receiving gossip", "err", err)
break
}
for k := range l {
delete(l, k)
}
err = g.onRecv(ctx, data, l)
if err != nil {
l["err"] = err
log.Debug("[Beacon Gossip] Recoverable Error", l)
}
}
}