2022-05-11 13:18:12 +00:00
|
|
|
//go:build go1.18
|
|
|
|
|
|
|
|
package sync
|
|
|
|
|
|
|
|
import (
|
2022-05-12 14:47:29 +00:00
|
|
|
"bytes"
|
2022-05-11 13:18:12 +00:00
|
|
|
"context"
|
2022-05-12 14:47:29 +00:00
|
|
|
"reflect"
|
2022-05-11 13:18:12 +00:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
pubsub "github.com/libp2p/go-libp2p-pubsub"
|
|
|
|
pb "github.com/libp2p/go-libp2p-pubsub/pb"
|
2022-10-07 07:24:51 +00:00
|
|
|
"github.com/libp2p/go-libp2p/core/peer"
|
2022-05-11 13:18:12 +00:00
|
|
|
gcache "github.com/patrickmn/go-cache"
|
2024-02-15 05:46:47 +00:00
|
|
|
mock "github.com/prysmaticlabs/prysm/v5/beacon-chain/blockchain/testing"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/core/helpers"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/core/signing"
|
|
|
|
dbtest "github.com/prysmaticlabs/prysm/v5/beacon-chain/db/testing"
|
|
|
|
doublylinkedtree "github.com/prysmaticlabs/prysm/v5/beacon-chain/forkchoice/doubly-linked-tree"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/p2p"
|
|
|
|
p2ptest "github.com/prysmaticlabs/prysm/v5/beacon-chain/p2p/testing"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/startup"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/state/stategen"
|
|
|
|
mockSync "github.com/prysmaticlabs/prysm/v5/beacon-chain/sync/initial-sync/testing"
|
|
|
|
lruwrpr "github.com/prysmaticlabs/prysm/v5/cache/lru"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/config/params"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/testing/assert"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/testing/require"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/testing/util"
|
2022-05-11 13:18:12 +00:00
|
|
|
)
|
|
|
|
|
2022-05-12 14:47:29 +00:00
|
|
|
func FuzzValidateBeaconBlockPubSub_Phase0(f *testing.F) {
|
|
|
|
db := dbtest.SetupDB(f)
|
|
|
|
p := p2ptest.NewFuzzTestP2P()
|
|
|
|
ctx := context.Background()
|
|
|
|
beaconState, privKeys := util.DeterministicGenesisState(f, 100)
|
|
|
|
parentBlock := util.NewBeaconBlock()
|
2022-06-24 17:22:39 +00:00
|
|
|
util.SaveBlock(f, ctx, db, parentBlock)
|
2022-05-12 14:47:29 +00:00
|
|
|
bRoot, err := parentBlock.Block.HashTreeRoot()
|
|
|
|
require.NoError(f, err)
|
|
|
|
require.NoError(f, db.SaveState(ctx, beaconState, bRoot))
|
|
|
|
require.NoError(f, db.SaveStateSummary(ctx, ðpb.StateSummary{Root: bRoot[:]}))
|
|
|
|
copied := beaconState.Copy()
|
|
|
|
require.NoError(f, copied.SetSlot(1))
|
|
|
|
proposerIdx, err := helpers.BeaconProposerIndex(ctx, copied)
|
|
|
|
require.NoError(f, err)
|
|
|
|
msg := util.NewBeaconBlock()
|
|
|
|
msg.Block.ParentRoot = bRoot[:]
|
|
|
|
msg.Block.Slot = 1
|
|
|
|
msg.Block.ProposerIndex = proposerIdx
|
|
|
|
msg.Signature, err = signing.ComputeDomainAndSign(beaconState, 0, msg.Block, params.BeaconConfig().DomainBeaconProposer, privKeys[proposerIdx])
|
|
|
|
require.NoError(f, err)
|
|
|
|
|
2022-09-28 20:10:27 +00:00
|
|
|
stateGen := stategen.New(db, doublylinkedtree.New())
|
2022-05-12 14:47:29 +00:00
|
|
|
chainService := &mock.ChainService{Genesis: time.Unix(time.Now().Unix()-int64(params.BeaconConfig().SecondsPerSlot), 0),
|
|
|
|
State: beaconState,
|
|
|
|
FinalizedCheckPoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 0,
|
|
|
|
Root: make([]byte, 32),
|
|
|
|
},
|
|
|
|
DB: db,
|
|
|
|
}
|
|
|
|
r := &Service{
|
|
|
|
cfg: &config{
|
|
|
|
beaconDB: db,
|
|
|
|
p2p: p,
|
|
|
|
initialSync: &mockSync.Sync{IsSyncing: false},
|
|
|
|
chain: chainService,
|
2023-05-03 04:34:01 +00:00
|
|
|
clock: startup.NewClock(chainService.Genesis, chainService.ValidatorsRoot),
|
2022-05-12 14:47:29 +00:00
|
|
|
blockNotifier: chainService.BlockNotifier(),
|
|
|
|
stateGen: stateGen,
|
|
|
|
},
|
|
|
|
seenBlockCache: lruwrpr.New(10),
|
|
|
|
badBlockCache: lruwrpr.New(10),
|
|
|
|
slotToPendingBlocks: gcache.New(time.Second, 2*time.Second),
|
|
|
|
seenPendingBlocks: make(map[[32]byte]bool),
|
|
|
|
}
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
_, err = p.Encoding().EncodeGossip(buf, msg)
|
|
|
|
require.NoError(f, err)
|
|
|
|
topic := p2p.GossipTypeMapping[reflect.TypeOf(msg)]
|
|
|
|
digest, err := r.currentForkDigest()
|
|
|
|
assert.NoError(f, err)
|
|
|
|
topic = r.addDigestToTopic(topic, digest)
|
|
|
|
|
|
|
|
f.Add("junk", []byte("junk"), buf.Bytes(), []byte(topic))
|
|
|
|
f.Fuzz(func(t *testing.T, pid string, from, data, topic []byte) {
|
|
|
|
r.cfg.p2p = p2ptest.NewFuzzTestP2P()
|
|
|
|
r.rateLimiter = newRateLimiter(r.cfg.p2p)
|
|
|
|
cService := &mock.ChainService{
|
|
|
|
Genesis: time.Unix(time.Now().Unix()-int64(params.BeaconConfig().SecondsPerSlot*10000000), 0),
|
|
|
|
State: beaconState,
|
|
|
|
FinalizedCheckPoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 0,
|
|
|
|
Root: make([]byte, 32),
|
|
|
|
},
|
|
|
|
DB: db,
|
|
|
|
}
|
|
|
|
r.cfg.chain = cService
|
|
|
|
r.cfg.blockNotifier = cService.BlockNotifier()
|
|
|
|
strTop := string(topic)
|
|
|
|
msg := &pubsub.Message{
|
|
|
|
Message: &pb.Message{
|
|
|
|
From: from,
|
|
|
|
Data: data,
|
|
|
|
Topic: &strTop,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
_, err := r.validateBeaconBlockPubSub(ctx, peer.ID(pid), msg)
|
|
|
|
_ = err
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func FuzzValidateBeaconBlockPubSub_Altair(f *testing.F) {
|
|
|
|
db := dbtest.SetupDB(f)
|
|
|
|
p := p2ptest.NewFuzzTestP2P()
|
|
|
|
ctx := context.Background()
|
|
|
|
beaconState, privKeys := util.DeterministicGenesisStateAltair(f, 100)
|
|
|
|
parentBlock := util.NewBeaconBlockAltair()
|
2022-06-24 17:22:39 +00:00
|
|
|
util.SaveBlock(f, ctx, db, parentBlock)
|
2022-05-12 14:47:29 +00:00
|
|
|
bRoot, err := parentBlock.Block.HashTreeRoot()
|
|
|
|
require.NoError(f, err)
|
|
|
|
require.NoError(f, db.SaveState(ctx, beaconState, bRoot))
|
|
|
|
require.NoError(f, db.SaveStateSummary(ctx, ðpb.StateSummary{Root: bRoot[:]}))
|
|
|
|
copied := beaconState.Copy()
|
|
|
|
require.NoError(f, copied.SetSlot(1))
|
|
|
|
proposerIdx, err := helpers.BeaconProposerIndex(ctx, copied)
|
|
|
|
require.NoError(f, err)
|
|
|
|
msg := util.NewBeaconBlock()
|
|
|
|
msg.Block.ParentRoot = bRoot[:]
|
|
|
|
msg.Block.Slot = 1
|
|
|
|
msg.Block.ProposerIndex = proposerIdx
|
|
|
|
msg.Signature, err = signing.ComputeDomainAndSign(beaconState, 0, msg.Block, params.BeaconConfig().DomainBeaconProposer, privKeys[proposerIdx])
|
|
|
|
require.NoError(f, err)
|
|
|
|
|
2022-09-28 20:10:27 +00:00
|
|
|
stateGen := stategen.New(db, doublylinkedtree.New())
|
2022-05-12 14:47:29 +00:00
|
|
|
chainService := &mock.ChainService{Genesis: time.Unix(time.Now().Unix()-int64(params.BeaconConfig().SecondsPerSlot), 0),
|
|
|
|
State: beaconState,
|
|
|
|
FinalizedCheckPoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 0,
|
|
|
|
Root: make([]byte, 32),
|
|
|
|
},
|
|
|
|
DB: db,
|
|
|
|
}
|
2022-05-11 13:18:12 +00:00
|
|
|
r := &Service{
|
|
|
|
cfg: &config{
|
2022-05-12 14:47:29 +00:00
|
|
|
beaconDB: db,
|
|
|
|
p2p: p,
|
|
|
|
initialSync: &mockSync.Sync{IsSyncing: false},
|
|
|
|
chain: chainService,
|
|
|
|
blockNotifier: chainService.BlockNotifier(),
|
|
|
|
stateGen: stateGen,
|
2023-05-03 04:34:01 +00:00
|
|
|
clock: startup.NewClock(chainService.Genesis, chainService.ValidatorsRoot),
|
2022-05-11 13:18:12 +00:00
|
|
|
},
|
2022-05-12 14:47:29 +00:00
|
|
|
seenBlockCache: lruwrpr.New(10),
|
|
|
|
badBlockCache: lruwrpr.New(10),
|
|
|
|
slotToPendingBlocks: gcache.New(time.Second, 2*time.Second),
|
|
|
|
seenPendingBlocks: make(map[[32]byte]bool),
|
2022-05-11 13:18:12 +00:00
|
|
|
}
|
2022-05-12 14:47:29 +00:00
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
_, err = p.Encoding().EncodeGossip(buf, msg)
|
|
|
|
require.NoError(f, err)
|
|
|
|
topic := p2p.GossipTypeMapping[reflect.TypeOf(msg)]
|
|
|
|
digest, err := r.currentForkDigest()
|
|
|
|
assert.NoError(f, err)
|
|
|
|
topic = r.addDigestToTopic(topic, digest)
|
|
|
|
|
|
|
|
f.Add("junk", []byte("junk"), buf.Bytes(), []byte(topic))
|
|
|
|
f.Fuzz(func(t *testing.T, pid string, from, data, topic []byte) {
|
2022-05-11 13:18:12 +00:00
|
|
|
r.cfg.p2p = p2ptest.NewFuzzTestP2P()
|
|
|
|
r.rateLimiter = newRateLimiter(r.cfg.p2p)
|
2022-05-12 14:47:29 +00:00
|
|
|
cService := &mock.ChainService{
|
|
|
|
Genesis: time.Unix(time.Now().Unix()-int64(params.BeaconConfig().SecondsPerSlot*10000000), 0),
|
|
|
|
State: beaconState,
|
|
|
|
FinalizedCheckPoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 0,
|
|
|
|
Root: make([]byte, 32),
|
|
|
|
},
|
|
|
|
DB: db,
|
|
|
|
}
|
|
|
|
r.cfg.chain = cService
|
|
|
|
r.cfg.blockNotifier = cService.BlockNotifier()
|
|
|
|
strTop := string(topic)
|
|
|
|
msg := &pubsub.Message{
|
|
|
|
Message: &pb.Message{
|
|
|
|
From: from,
|
|
|
|
Data: data,
|
|
|
|
Topic: &strTop,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
_, err := r.validateBeaconBlockPubSub(ctx, peer.ID(pid), msg)
|
|
|
|
_ = err
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func FuzzValidateBeaconBlockPubSub_Bellatrix(f *testing.F) {
|
|
|
|
db := dbtest.SetupDB(f)
|
|
|
|
p := p2ptest.NewFuzzTestP2P()
|
|
|
|
ctx := context.Background()
|
|
|
|
beaconState, privKeys := util.DeterministicGenesisStateBellatrix(f, 100)
|
|
|
|
parentBlock := util.NewBeaconBlockBellatrix()
|
2022-06-24 17:22:39 +00:00
|
|
|
util.SaveBlock(f, ctx, db, parentBlock)
|
2022-05-12 14:47:29 +00:00
|
|
|
bRoot, err := parentBlock.Block.HashTreeRoot()
|
|
|
|
require.NoError(f, err)
|
|
|
|
require.NoError(f, db.SaveState(ctx, beaconState, bRoot))
|
|
|
|
require.NoError(f, db.SaveStateSummary(ctx, ðpb.StateSummary{Root: bRoot[:]}))
|
|
|
|
copied := beaconState.Copy()
|
|
|
|
require.NoError(f, copied.SetSlot(1))
|
|
|
|
proposerIdx, err := helpers.BeaconProposerIndex(ctx, copied)
|
|
|
|
require.NoError(f, err)
|
|
|
|
msg := util.NewBeaconBlock()
|
|
|
|
msg.Block.ParentRoot = bRoot[:]
|
|
|
|
msg.Block.Slot = 1
|
|
|
|
msg.Block.ProposerIndex = proposerIdx
|
|
|
|
msg.Signature, err = signing.ComputeDomainAndSign(beaconState, 0, msg.Block, params.BeaconConfig().DomainBeaconProposer, privKeys[proposerIdx])
|
|
|
|
require.NoError(f, err)
|
|
|
|
|
2022-09-28 20:10:27 +00:00
|
|
|
stateGen := stategen.New(db, doublylinkedtree.New())
|
2022-05-12 14:47:29 +00:00
|
|
|
chainService := &mock.ChainService{Genesis: time.Unix(time.Now().Unix()-int64(params.BeaconConfig().SecondsPerSlot), 0),
|
|
|
|
State: beaconState,
|
|
|
|
FinalizedCheckPoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 0,
|
|
|
|
Root: make([]byte, 32),
|
|
|
|
},
|
|
|
|
DB: db,
|
|
|
|
}
|
|
|
|
r := &Service{
|
|
|
|
cfg: &config{
|
|
|
|
beaconDB: db,
|
|
|
|
p2p: p,
|
|
|
|
initialSync: &mockSync.Sync{IsSyncing: false},
|
|
|
|
chain: chainService,
|
2023-05-03 04:34:01 +00:00
|
|
|
clock: startup.NewClock(chainService.Genesis, chainService.ValidatorsRoot),
|
2022-05-12 14:47:29 +00:00
|
|
|
blockNotifier: chainService.BlockNotifier(),
|
|
|
|
stateGen: stateGen,
|
|
|
|
},
|
|
|
|
seenBlockCache: lruwrpr.New(10),
|
|
|
|
badBlockCache: lruwrpr.New(10),
|
|
|
|
slotToPendingBlocks: gcache.New(time.Second, 2*time.Second),
|
|
|
|
seenPendingBlocks: make(map[[32]byte]bool),
|
|
|
|
}
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
_, err = p.Encoding().EncodeGossip(buf, msg)
|
|
|
|
require.NoError(f, err)
|
|
|
|
topic := p2p.GossipTypeMapping[reflect.TypeOf(msg)]
|
|
|
|
digest, err := r.currentForkDigest()
|
|
|
|
assert.NoError(f, err)
|
|
|
|
topic = r.addDigestToTopic(topic, digest)
|
|
|
|
|
|
|
|
f.Add("junk", []byte("junk"), buf.Bytes(), []byte(topic))
|
|
|
|
f.Fuzz(func(t *testing.T, pid string, from, data, topic []byte) {
|
|
|
|
r.cfg.p2p = p2ptest.NewFuzzTestP2P()
|
|
|
|
r.rateLimiter = newRateLimiter(r.cfg.p2p)
|
|
|
|
cService := &mock.ChainService{
|
|
|
|
Genesis: time.Unix(time.Now().Unix()-int64(params.BeaconConfig().SecondsPerSlot*10000000), 0),
|
|
|
|
State: beaconState,
|
|
|
|
FinalizedCheckPoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 0,
|
|
|
|
Root: make([]byte, 32),
|
|
|
|
},
|
|
|
|
DB: db,
|
|
|
|
}
|
|
|
|
r.cfg.chain = cService
|
|
|
|
r.cfg.blockNotifier = cService.BlockNotifier()
|
2022-05-11 13:18:12 +00:00
|
|
|
strTop := string(topic)
|
|
|
|
msg := &pubsub.Message{
|
|
|
|
Message: &pb.Message{
|
2022-05-12 14:47:29 +00:00
|
|
|
From: from,
|
|
|
|
Data: data,
|
|
|
|
Topic: &strTop,
|
2022-05-11 13:18:12 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
_, err := r.validateBeaconBlockPubSub(ctx, peer.ID(pid), msg)
|
|
|
|
_ = err
|
|
|
|
})
|
|
|
|
}
|