2022-11-25 15:38:22 +00:00
|
|
|
package network
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
2022-11-28 23:00:40 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/gointerfaces/sentinel"
|
2023-01-17 01:22:05 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/clparams"
|
2022-11-25 15:38:22 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/cltypes"
|
2023-03-12 14:41:53 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/cltypes/ssz"
|
2023-01-11 16:18:14 +00:00
|
|
|
"github.com/ledgerwatch/erigon/common"
|
2022-11-25 15:38:22 +00:00
|
|
|
"github.com/ledgerwatch/log/v3"
|
|
|
|
)
|
|
|
|
|
|
|
|
type GossipReceiver interface {
|
2023-03-12 14:41:53 +00:00
|
|
|
ReceiveGossip(ssz.Unmarshaler)
|
2022-11-25 15:38:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type GossipManager struct {
|
|
|
|
ctx context.Context
|
|
|
|
|
2022-11-28 23:00:40 +00:00
|
|
|
receivers map[sentinel.GossipType][]GossipReceiver
|
|
|
|
sentinel sentinel.SentinelClient
|
2022-11-25 15:38:22 +00:00
|
|
|
}
|
|
|
|
|
2022-11-28 23:00:40 +00:00
|
|
|
func NewGossipReceiver(ctx context.Context, s sentinel.SentinelClient) *GossipManager {
|
2022-11-25 15:38:22 +00:00
|
|
|
return &GossipManager{
|
2022-12-03 02:16:26 +00:00
|
|
|
sentinel: s,
|
2022-11-28 23:00:40 +00:00
|
|
|
receivers: make(map[sentinel.GossipType][]GossipReceiver),
|
2022-12-03 02:16:26 +00:00
|
|
|
ctx: ctx,
|
2022-11-25 15:38:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-28 23:00:40 +00:00
|
|
|
func (g *GossipManager) AddReceiver(t sentinel.GossipType, receiver GossipReceiver) {
|
2022-11-25 15:38:22 +00:00
|
|
|
if _, ok := g.receivers[t]; !ok {
|
|
|
|
g.receivers[t] = make([]GossipReceiver, 0)
|
|
|
|
}
|
|
|
|
g.receivers[t] = append(g.receivers[t], receiver)
|
|
|
|
}
|
|
|
|
|
2022-12-03 02:16:26 +00:00
|
|
|
func (g *GossipManager) Loop() {
|
2022-12-05 00:25:12 +00:00
|
|
|
subscription, err := g.sentinel.SubscribeGossip(g.ctx, &sentinel.EmptyMessage{})
|
2022-11-25 15:38:22 +00:00
|
|
|
if err != nil {
|
2022-12-03 02:16:26 +00:00
|
|
|
return
|
2022-11-25 15:38:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for {
|
|
|
|
data, err := subscription.Recv()
|
|
|
|
if err != nil {
|
|
|
|
log.Warn("[Beacon Gossip] Failure in receiving", "err", err)
|
|
|
|
continue
|
|
|
|
}
|
2022-12-21 18:47:29 +00:00
|
|
|
// 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 continues to the next iteration.
|
|
|
|
//If the deserialization is successful, the object is set to the deserialized value and the loop continues to the next iteration.
|
2022-11-25 15:38:22 +00:00
|
|
|
receivers := g.receivers[data.Type]
|
2023-03-12 14:41:53 +00:00
|
|
|
var object ssz.Unmarshaler
|
2022-11-25 15:38:22 +00:00
|
|
|
switch data.Type {
|
2022-11-28 23:00:40 +00:00
|
|
|
case sentinel.GossipType_BeaconBlockGossipType:
|
2023-01-17 01:22:05 +00:00
|
|
|
object = &cltypes.SignedBeaconBlock{}
|
2023-01-21 21:33:50 +00:00
|
|
|
if err := object.DecodeSSZWithVersion(common.CopyBytes(data.Data), int(clparams.BellatrixVersion)); err != nil {
|
2023-03-19 20:13:53 +00:00
|
|
|
log.Debug("[Beacon Gossip] Failure in decoding block", "err", err)
|
|
|
|
g.sentinel.BanPeer(g.ctx, data.Peer)
|
2022-11-25 15:38:22 +00:00
|
|
|
continue
|
|
|
|
}
|
2022-11-28 23:00:40 +00:00
|
|
|
case sentinel.GossipType_VoluntaryExitGossipType:
|
2023-01-17 01:22:05 +00:00
|
|
|
object = &cltypes.SignedVoluntaryExit{}
|
2023-01-21 21:33:50 +00:00
|
|
|
if err := object.DecodeSSZWithVersion(data.Data, int(clparams.BellatrixVersion)); err != nil {
|
2023-03-19 20:13:53 +00:00
|
|
|
log.Debug("[Beacon Gossip] Failure in decoding exit", "err", err)
|
|
|
|
g.sentinel.BanPeer(g.ctx, data.Peer)
|
2022-11-25 15:38:22 +00:00
|
|
|
continue
|
|
|
|
}
|
2022-11-28 23:00:40 +00:00
|
|
|
case sentinel.GossipType_ProposerSlashingGossipType:
|
2022-11-25 15:38:22 +00:00
|
|
|
object = &cltypes.ProposerSlashing{}
|
2023-01-21 21:33:50 +00:00
|
|
|
if err := object.DecodeSSZWithVersion(data.Data, int(clparams.BellatrixVersion)); err != nil {
|
2023-03-19 20:13:53 +00:00
|
|
|
log.Debug("[Beacon Gossip] Failure in decoding proposer slashing", "err", err)
|
|
|
|
g.sentinel.BanPeer(g.ctx, data.Peer)
|
2022-11-25 15:38:22 +00:00
|
|
|
continue
|
|
|
|
}
|
2022-11-28 23:00:40 +00:00
|
|
|
case sentinel.GossipType_AttesterSlashingGossipType:
|
2022-11-25 15:38:22 +00:00
|
|
|
object = &cltypes.AttesterSlashing{}
|
2023-01-21 21:33:50 +00:00
|
|
|
if err := object.DecodeSSZWithVersion(data.Data, int(clparams.BellatrixVersion)); err != nil {
|
2023-03-19 20:13:53 +00:00
|
|
|
log.Debug("[Beacon Gossip] Failure in decoding attester slashing", "err", err)
|
|
|
|
g.sentinel.BanPeer(g.ctx, data.Peer)
|
2022-11-25 15:38:22 +00:00
|
|
|
continue
|
|
|
|
}
|
2022-11-28 23:00:40 +00:00
|
|
|
case sentinel.GossipType_AggregateAndProofGossipType:
|
2022-12-03 02:16:26 +00:00
|
|
|
object = &cltypes.SignedAggregateAndProof{}
|
2023-01-21 21:33:50 +00:00
|
|
|
if err := object.DecodeSSZWithVersion(data.Data, int(clparams.BellatrixVersion)); err != nil {
|
2023-03-19 20:13:53 +00:00
|
|
|
log.Debug("[Beacon Gossip] Failure in decoding proof", "err", err)
|
|
|
|
g.sentinel.BanPeer(g.ctx, data.Peer)
|
2022-11-25 15:38:22 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// If we received a valid object give it to our receiver
|
|
|
|
if object != nil {
|
|
|
|
for _, receiver := range receivers {
|
|
|
|
receiver.ReceiveGossip(object)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|