mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-09 20:41:20 +00:00
96 lines
2.7 KiB
Go
96 lines
2.7 KiB
Go
|
package network
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
|
||
|
"github.com/ledgerwatch/erigon/cl/cltypes"
|
||
|
"github.com/ledgerwatch/erigon/cl/rpc/consensusrpc"
|
||
|
"github.com/ledgerwatch/log/v3"
|
||
|
)
|
||
|
|
||
|
type GossipReceiver interface {
|
||
|
ReceiveGossip(cltypes.ObjectSSZ)
|
||
|
}
|
||
|
|
||
|
type GossipManager struct {
|
||
|
ctx context.Context
|
||
|
|
||
|
receivers map[consensusrpc.GossipType][]GossipReceiver
|
||
|
sentinel consensusrpc.SentinelClient
|
||
|
}
|
||
|
|
||
|
func NewGossipReceiver(ctx context.Context, sentinel consensusrpc.SentinelClient) *GossipManager {
|
||
|
return &GossipManager{
|
||
|
sentinel: sentinel,
|
||
|
|
||
|
receivers: make(map[consensusrpc.GossipType][]GossipReceiver),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (g *GossipManager) AddReceiver(t consensusrpc.GossipType, receiver GossipReceiver) {
|
||
|
if _, ok := g.receivers[t]; !ok {
|
||
|
g.receivers[t] = make([]GossipReceiver, 0)
|
||
|
}
|
||
|
g.receivers[t] = append(g.receivers[t], receiver)
|
||
|
}
|
||
|
|
||
|
func (g *GossipManager) Start() error {
|
||
|
subscription, err := g.sentinel.SubscribeGossip(g.ctx, &consensusrpc.EmptyRequest{})
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
go g.loop(subscription)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (g *GossipManager) loop(subscription consensusrpc.Sentinel_SubscribeGossipClient) {
|
||
|
for {
|
||
|
data, err := subscription.Recv()
|
||
|
if err != nil {
|
||
|
log.Warn("[Beacon Gossip] Failure in receiving", "err", err)
|
||
|
continue
|
||
|
}
|
||
|
receivers := g.receivers[data.Type]
|
||
|
var object cltypes.ObjectSSZ
|
||
|
switch data.Type {
|
||
|
case consensusrpc.GossipType_BeaconBlockGossipType:
|
||
|
object = &cltypes.SignedBeaconBlockBellatrix{}
|
||
|
if err := object.UnmarshalSSZ(data.Data); err != nil {
|
||
|
log.Warn("[Beacon Gossip] Failure in decoding block", "err", err)
|
||
|
continue
|
||
|
}
|
||
|
case consensusrpc.GossipType_VoluntaryExitGossipType:
|
||
|
object = &cltypes.VoluntaryExit{}
|
||
|
if err := object.UnmarshalSSZ(data.Data); err != nil {
|
||
|
log.Warn("[Beacon Gossip] Failure in decoding block", "err", err)
|
||
|
continue
|
||
|
}
|
||
|
case consensusrpc.GossipType_ProposerSlashingGossipType:
|
||
|
object = &cltypes.ProposerSlashing{}
|
||
|
if err := object.UnmarshalSSZ(data.Data); err != nil {
|
||
|
log.Warn("[Beacon Gossip] Failure in decoding block", "err", err)
|
||
|
continue
|
||
|
}
|
||
|
case consensusrpc.GossipType_AttesterSlashingGossipType:
|
||
|
object = &cltypes.AttesterSlashing{}
|
||
|
if err := object.UnmarshalSSZ(data.Data); err != nil {
|
||
|
log.Warn("[Beacon Gossip] Failure in decoding block", "err", err)
|
||
|
continue
|
||
|
}
|
||
|
case consensusrpc.GossipType_AggregateAndProofGossipType:
|
||
|
object = &cltypes.AggregateAndProof{}
|
||
|
if err := object.UnmarshalSSZ(data.Data); err != nil {
|
||
|
log.Warn("[Beacon Gossip] Failure in decoding block", "err", err)
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
}
|
||
|
// If we received a valid object give it to our receiver
|
||
|
if object != nil {
|
||
|
for _, receiver := range receivers {
|
||
|
receiver.ReceiveGossip(object)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|