mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-03 09:37:38 +00:00
162 lines
4.3 KiB
Go
162 lines
4.3 KiB
Go
package persistence
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
"github.com/ledgerwatch/erigon/cl/cltypes"
|
|
"github.com/ledgerwatch/erigon/cl/phase1/network"
|
|
"github.com/ledgerwatch/erigon/cl/rpc"
|
|
"github.com/ledgerwatch/erigon/cl/sentinel/peers"
|
|
"github.com/tidwall/btree"
|
|
)
|
|
|
|
var _ BlockSource = (*BeaconRpcSource)(nil)
|
|
|
|
type BeaconRpcSource struct {
|
|
rpc *rpc.BeaconRpcP2P
|
|
}
|
|
|
|
func (b *BeaconRpcSource) SaveBlocks(ctx context.Context, blocks *peers.PeeredObject[*cltypes.SignedBeaconBlock]) error {
|
|
// it is a no-op because there is no need to do this
|
|
return nil
|
|
}
|
|
|
|
func NewBeaconRpcSource(rpc *rpc.BeaconRpcP2P) *BeaconRpcSource {
|
|
return &BeaconRpcSource{
|
|
rpc: rpc,
|
|
}
|
|
}
|
|
|
|
func (*BeaconRpcSource) GetBlock(ctx context.Context, tx kv.Tx, slot uint64) (*peers.PeeredObject[*cltypes.SignedBeaconBlock], error) {
|
|
panic("unimplemented")
|
|
}
|
|
|
|
func (b *BeaconRpcSource) GetRange(ctx context.Context, _ kv.Tx, from uint64, count uint64) (*peers.PeeredObject[[]*cltypes.SignedBeaconBlock], error) {
|
|
if count == 0 {
|
|
return nil, nil
|
|
}
|
|
var responses *peers.PeeredObject[[]*cltypes.SignedBeaconBlock]
|
|
reqInterval := time.NewTicker(200 * time.Millisecond)
|
|
doneRespCh := make(chan *peers.PeeredObject[[]*cltypes.SignedBeaconBlock], 1)
|
|
defer reqInterval.Stop()
|
|
|
|
for {
|
|
select {
|
|
case <-reqInterval.C:
|
|
go func() {
|
|
responses, pid, err := b.rpc.SendBeaconBlocksByRangeReq(ctx, from, count)
|
|
if err != nil {
|
|
return
|
|
}
|
|
select {
|
|
case doneRespCh <- &peers.PeeredObject[[]*cltypes.SignedBeaconBlock]{Data: responses, Peer: pid}:
|
|
default:
|
|
}
|
|
}()
|
|
case <-ctx.Done():
|
|
return nil, ctx.Err()
|
|
case responses = <-doneRespCh:
|
|
return responses, nil
|
|
}
|
|
}
|
|
}
|
|
|
|
// a noop for rpc source since we always return new data
|
|
func (b *BeaconRpcSource) PurgeRange(ctx context.Context, _ kv.Tx, from uint64, count uint64) error {
|
|
return nil
|
|
}
|
|
|
|
var _ BlockSource = (*GossipSource)(nil)
|
|
|
|
type GossipSource struct {
|
|
gossip *network.GossipManager
|
|
gossipBlocks <-chan *peers.PeeredObject[*cltypes.SignedBeaconBlock]
|
|
|
|
mu sync.Mutex
|
|
blocks *btree.Map[uint64, chan *peers.PeeredObject[*cltypes.SignedBeaconBlock]]
|
|
}
|
|
|
|
func (*GossipSource) GetBlock(ctx context.Context, tx kv.Tx, slot uint64) (*peers.PeeredObject[*cltypes.SignedBeaconBlock], error) {
|
|
panic("unimplemented")
|
|
}
|
|
|
|
func NewGossipSource(ctx context.Context, gossip *network.GossipManager) *GossipSource {
|
|
g := &GossipSource{
|
|
gossip: gossip,
|
|
gossipBlocks: gossip.SubscribeSignedBeaconBlocks(ctx),
|
|
blocks: btree.NewMap[uint64, chan *peers.PeeredObject[*cltypes.SignedBeaconBlock]](32),
|
|
}
|
|
go func() {
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case recv := <-g.gossipBlocks:
|
|
ch := g.grabOrCreate(ctx, recv.Data.Block.Slot)
|
|
select {
|
|
case ch <- recv:
|
|
default:
|
|
}
|
|
}
|
|
}
|
|
}()
|
|
return g
|
|
}
|
|
|
|
func (b *GossipSource) grabOrCreate(ctx context.Context, id uint64) chan *peers.PeeredObject[*cltypes.SignedBeaconBlock] {
|
|
b.mu.Lock()
|
|
defer b.mu.Unlock()
|
|
ch, ok := b.blocks.Get(id)
|
|
if !ok {
|
|
ch = make(chan *peers.PeeredObject[*cltypes.SignedBeaconBlock], 3)
|
|
b.blocks.Set(id, ch)
|
|
}
|
|
// if there are ever more than 512 blocks, clear the last 256 blocks
|
|
if b.blocks.Len() > 512 {
|
|
b.purgeRange(ctx, nil, 0, id-256)
|
|
}
|
|
return ch
|
|
}
|
|
func (b *GossipSource) GetRange(ctx context.Context, _ kv.Tx, from uint64, count uint64) (*peers.PeeredObject[[]*cltypes.SignedBeaconBlock], error) {
|
|
out := &peers.PeeredObject[[]*cltypes.SignedBeaconBlock]{}
|
|
for i := from; i < from+count; i++ {
|
|
ch := b.grabOrCreate(ctx, i)
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil, ctx.Err()
|
|
case item := <-ch:
|
|
out.Data = append(out.Data, item.Data)
|
|
out.Peer = item.Peer
|
|
}
|
|
}
|
|
return out, nil
|
|
}
|
|
|
|
func (b *GossipSource) PurgeRange(ctx context.Context, tx kv.Tx, from uint64, count uint64) error {
|
|
b.mu.Lock()
|
|
defer b.mu.Unlock()
|
|
return b.purgeRange(ctx, tx, from, count)
|
|
}
|
|
|
|
func (b *GossipSource) purgeRange(ctx context.Context, _ kv.Tx, from uint64, count uint64) error {
|
|
initSize := count
|
|
if initSize > 256 {
|
|
initSize = 256
|
|
}
|
|
xs := make([]uint64, 0, initSize)
|
|
b.blocks.Ascend(from, func(key uint64, value chan *peers.PeeredObject[*cltypes.SignedBeaconBlock]) bool {
|
|
if key >= from+count {
|
|
return false
|
|
}
|
|
xs = append(xs, key)
|
|
return true
|
|
})
|
|
for _, v := range xs {
|
|
b.blocks.Delete(v)
|
|
}
|
|
return nil
|
|
}
|