2022-05-03 20:59:07 +00:00
|
|
|
package forkchoice
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-09-15 05:42:20 +00:00
|
|
|
"errors"
|
2022-05-03 20:59:07 +00:00
|
|
|
"fmt"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
2023-03-17 18:52:56 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/blockchain"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/execution"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/interfaces"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/testing/require"
|
2022-05-03 20:59:07 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type Builder struct {
|
|
|
|
service *blockchain.Service
|
|
|
|
lastTick int64
|
|
|
|
execMock *engineMock
|
|
|
|
}
|
|
|
|
|
2023-02-09 09:23:32 +00:00
|
|
|
func NewBuilder(t testing.TB, initialState state.BeaconState, initialBlock interfaces.ReadOnlySignedBeaconBlock) *Builder {
|
2022-05-03 20:59:07 +00:00
|
|
|
execMock := &engineMock{
|
|
|
|
powBlocks: make(map[[32]byte]*ethpb.PowBlock),
|
|
|
|
}
|
|
|
|
service := startChainService(t, initialState, initialBlock, execMock)
|
|
|
|
return &Builder{
|
|
|
|
service: service,
|
|
|
|
execMock: execMock,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Tick resets the genesis time to now()-tick and adjusts the slot to the appropriate value.
|
|
|
|
func (bb *Builder) Tick(t testing.TB, tick int64) {
|
2023-03-21 18:39:40 +00:00
|
|
|
bb.service.ForkChoicer().Lock()
|
|
|
|
defer bb.service.ForkChoicer().Unlock()
|
|
|
|
|
2022-05-03 20:59:07 +00:00
|
|
|
bb.service.SetGenesisTime(time.Unix(time.Now().Unix()-tick, 0))
|
2022-08-19 14:25:35 +00:00
|
|
|
lastSlot := uint64(bb.lastTick) / params.BeaconConfig().SecondsPerSlot
|
|
|
|
currentSlot := uint64(tick) / params.BeaconConfig().SecondsPerSlot
|
|
|
|
for lastSlot < currentSlot {
|
|
|
|
lastSlot++
|
|
|
|
bb.service.ForkChoicer().SetGenesisTime(uint64(time.Now().Unix() - int64(params.BeaconConfig().SecondsPerSlot*lastSlot)))
|
2023-01-26 14:40:12 +00:00
|
|
|
require.NoError(t, bb.service.ForkChoicer().NewSlot(context.TODO(), primitives.Slot(lastSlot)))
|
2022-05-03 20:59:07 +00:00
|
|
|
}
|
2022-08-19 14:25:35 +00:00
|
|
|
if tick > int64(params.BeaconConfig().SecondsPerSlot*lastSlot) {
|
|
|
|
bb.service.ForkChoicer().SetGenesisTime(uint64(time.Now().Unix() - tick))
|
|
|
|
}
|
|
|
|
bb.lastTick = tick
|
2022-05-03 20:59:07 +00:00
|
|
|
}
|
|
|
|
|
2022-09-15 05:42:20 +00:00
|
|
|
// SetPayloadStatus sets the payload status that the engine will return
|
|
|
|
func (bb *Builder) SetPayloadStatus(resp *MockEngineResp) error {
|
|
|
|
if resp == nil {
|
|
|
|
return errors.New("invalid nil payload status")
|
|
|
|
}
|
|
|
|
if resp.LatestValidHash == nil {
|
|
|
|
bb.execMock.latestValidHash = common.FromHex("0x0000000000000000000000000000000000000000000000000000000000000000")
|
|
|
|
} else {
|
|
|
|
bb.execMock.latestValidHash = common.FromHex(*resp.LatestValidHash)
|
|
|
|
}
|
|
|
|
if resp.Status == nil {
|
|
|
|
return errors.New("invalid nil status")
|
|
|
|
}
|
|
|
|
switch *resp.Status {
|
|
|
|
case "SYNCING":
|
|
|
|
bb.execMock.payloadStatus = execution.ErrAcceptedSyncingPayloadStatus
|
|
|
|
case "VALID":
|
|
|
|
bb.execMock.payloadStatus = nil
|
|
|
|
case "INVALID":
|
|
|
|
bb.execMock.payloadStatus = execution.ErrInvalidPayloadStatus
|
|
|
|
default:
|
|
|
|
return errors.New("unknown payload status")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-06-22 13:56:07 +00:00
|
|
|
// block returns the block root.
|
2023-02-09 09:23:32 +00:00
|
|
|
func (bb *Builder) block(t testing.TB, b interfaces.ReadOnlySignedBeaconBlock) [32]byte {
|
2022-05-03 20:59:07 +00:00
|
|
|
r, err := b.Block().HashTreeRoot()
|
|
|
|
require.NoError(t, err)
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
|
|
|
// InvalidBlock receives the invalid block and notifies forkchoice.
|
2023-02-09 09:23:32 +00:00
|
|
|
func (bb *Builder) InvalidBlock(t testing.TB, b interfaces.ReadOnlySignedBeaconBlock) {
|
2022-05-03 20:59:07 +00:00
|
|
|
r := bb.block(t, b)
|
|
|
|
require.Equal(t, true, bb.service.ReceiveBlock(context.TODO(), b, r) != nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ValidBlock receives the valid block and notifies forkchoice.
|
2023-02-09 09:23:32 +00:00
|
|
|
func (bb *Builder) ValidBlock(t testing.TB, b interfaces.ReadOnlySignedBeaconBlock) {
|
2022-05-03 20:59:07 +00:00
|
|
|
r := bb.block(t, b)
|
|
|
|
require.NoError(t, bb.service.ReceiveBlock(context.TODO(), b, r))
|
|
|
|
}
|
|
|
|
|
|
|
|
// PoWBlock receives the block and notifies a mocked execution engine.
|
2022-06-27 13:34:38 +00:00
|
|
|
func (bb *Builder) PoWBlock(pb *ethpb.PowBlock) {
|
2022-05-03 20:59:07 +00:00
|
|
|
bb.execMock.powBlocks[bytesutil.ToBytes32(pb.BlockHash)] = pb
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attestation receives the attestation and updates forkchoice.
|
|
|
|
func (bb *Builder) Attestation(t testing.TB, a *ethpb.Attestation) {
|
2023-03-21 18:39:40 +00:00
|
|
|
bb.service.ForkChoicer().Lock()
|
|
|
|
defer bb.service.ForkChoicer().Unlock()
|
|
|
|
|
2023-03-04 23:19:23 +00:00
|
|
|
require.NoError(t, bb.service.OnAttestation(context.TODO(), a, params.BeaconNetworkConfig().MaximumGossipClockDisparity))
|
2022-05-03 20:59:07 +00:00
|
|
|
}
|
|
|
|
|
2022-05-23 23:08:24 +00:00
|
|
|
// AttesterSlashing receives an attester slashing and feeds it to forkchoice.
|
|
|
|
func (bb *Builder) AttesterSlashing(s *ethpb.AttesterSlashing) {
|
2023-03-21 18:39:40 +00:00
|
|
|
bb.service.ForkChoicer().Lock()
|
|
|
|
defer bb.service.ForkChoicer().Unlock()
|
|
|
|
|
2022-05-23 23:08:24 +00:00
|
|
|
slashings := []*ethpb.AttesterSlashing{s}
|
|
|
|
bb.service.InsertSlashingsToForkChoiceStore(context.TODO(), slashings)
|
|
|
|
}
|
|
|
|
|
2022-05-03 20:59:07 +00:00
|
|
|
// Check evaluates the fork choice results and compares them to the expected values.
|
|
|
|
func (bb *Builder) Check(t testing.TB, c *Check) {
|
|
|
|
if c == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx := context.TODO()
|
2023-03-21 18:39:40 +00:00
|
|
|
bb.service.ForkChoicer().Lock()
|
2022-05-03 20:59:07 +00:00
|
|
|
require.NoError(t, bb.service.UpdateAndSaveHeadWithBalances(ctx))
|
2023-03-21 18:39:40 +00:00
|
|
|
bb.service.ForkChoicer().Unlock()
|
2022-05-03 20:59:07 +00:00
|
|
|
if c.Head != nil {
|
|
|
|
r, err := bb.service.HeadRoot(ctx)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.DeepEqual(t, common.FromHex(c.Head.Root), r)
|
2023-01-26 14:40:12 +00:00
|
|
|
require.Equal(t, primitives.Slot(c.Head.Slot), bb.service.HeadSlot())
|
2022-05-03 20:59:07 +00:00
|
|
|
}
|
|
|
|
if c.JustifiedCheckPoint != nil {
|
|
|
|
cp := ðpb.Checkpoint{
|
2023-01-26 14:40:12 +00:00
|
|
|
Epoch: primitives.Epoch(c.JustifiedCheckPoint.Epoch),
|
2022-05-03 20:59:07 +00:00
|
|
|
Root: common.FromHex(c.JustifiedCheckPoint.Root),
|
|
|
|
}
|
2022-06-25 03:57:52 +00:00
|
|
|
got := bb.service.CurrentJustifiedCheckpt()
|
2022-05-20 18:41:33 +00:00
|
|
|
require.DeepEqual(t, cp, got)
|
2022-05-03 20:59:07 +00:00
|
|
|
}
|
|
|
|
if c.FinalizedCheckPoint != nil {
|
|
|
|
cp := ðpb.Checkpoint{
|
2023-01-26 14:40:12 +00:00
|
|
|
Epoch: primitives.Epoch(c.FinalizedCheckPoint.Epoch),
|
2022-05-03 20:59:07 +00:00
|
|
|
Root: common.FromHex(c.FinalizedCheckPoint.Root),
|
|
|
|
}
|
2022-06-25 03:57:52 +00:00
|
|
|
got := bb.service.FinalizedCheckpt()
|
2022-05-20 18:41:33 +00:00
|
|
|
require.DeepSSZEqual(t, cp, got)
|
2022-05-03 20:59:07 +00:00
|
|
|
}
|
|
|
|
if c.ProposerBoostRoot != nil {
|
|
|
|
want := fmt.Sprintf("%#x", common.FromHex(*c.ProposerBoostRoot))
|
2023-03-21 18:39:40 +00:00
|
|
|
bb.service.ForkChoicer().RLock()
|
|
|
|
got := fmt.Sprintf("%#x", bb.service.ForkChoicer().ProposerBoost())
|
|
|
|
bb.service.ForkChoicer().RUnlock()
|
2022-05-03 20:59:07 +00:00
|
|
|
require.DeepEqual(t, want, got)
|
|
|
|
}
|
2022-05-22 18:37:01 +00:00
|
|
|
|
2022-05-03 20:59:07 +00:00
|
|
|
}
|