package forkchoice import ( "context" "fmt" "math/big" "path" "testing" "time" "github.com/ethereum/go-ethereum/common" "github.com/golang/snappy" forkchoicetypes "github.com/prysmaticlabs/prysm/beacon-chain/forkchoice/types" "github.com/prysmaticlabs/prysm/beacon-chain/state" v1 "github.com/prysmaticlabs/prysm/beacon-chain/state/v1" v2 "github.com/prysmaticlabs/prysm/beacon-chain/state/v2" v3 "github.com/prysmaticlabs/prysm/beacon-chain/state/v3" fieldparams "github.com/prysmaticlabs/prysm/config/fieldparams" "github.com/prysmaticlabs/prysm/config/params" "github.com/prysmaticlabs/prysm/consensus-types/interfaces" types "github.com/prysmaticlabs/prysm/consensus-types/primitives" "github.com/prysmaticlabs/prysm/consensus-types/wrapper" "github.com/prysmaticlabs/prysm/encoding/bytesutil" ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/runtime/version" "github.com/prysmaticlabs/prysm/testing/require" "github.com/prysmaticlabs/prysm/testing/spectest/utils" "github.com/prysmaticlabs/prysm/testing/util" "github.com/prysmaticlabs/prysm/time/slots" ) // Run executes "forkchoice" test. func Run(t *testing.T, config string, fork int) { require.NoError(t, utils.SetConfig(t, config)) testFolders, _ := utils.TestFolders(t, config, version.String(fork), "fork_choice") for _, folder := range testFolders { folderPath := path.Join("fork_choice", folder.Name(), "pyspec_tests") testFolders, testsFolderPath := utils.TestFolders(t, config, version.String(fork), folderPath) for _, folder := range testFolders { t.Run(folder.Name(), func(t *testing.T) { ctx := context.Background() preStepsFile, err := util.BazelFileBytes(testsFolderPath, folder.Name(), "steps.yaml") require.NoError(t, err) var steps []Step require.NoError(t, utils.UnmarshalYaml(preStepsFile, &steps)) preBeaconStateFile, err := util.BazelFileBytes(testsFolderPath, folder.Name(), "anchor_state.ssz_snappy") require.NoError(t, err) preBeaconStateSSZ, err := snappy.Decode(nil /* dst */, preBeaconStateFile) require.NoError(t, err) blockFile, err := util.BazelFileBytes(testsFolderPath, folder.Name(), "anchor_block.ssz_snappy") require.NoError(t, err) blockSSZ, err := snappy.Decode(nil /* dst */, blockFile) require.NoError(t, err) var beaconState state.BeaconState var beaconBlock interfaces.SignedBeaconBlock switch fork { case version.Phase0: beaconState = unmarshalPhase0State(t, preBeaconStateSSZ) beaconBlock = unmarshalPhase0Block(t, blockSSZ) case version.Altair: beaconState = unmarshalAltairState(t, preBeaconStateSSZ) beaconBlock = unmarshalAltairBlock(t, blockSSZ) case version.Bellatrix: beaconState = unmarshalBellatrixState(t, preBeaconStateSSZ) beaconBlock = unmarshalBellatrixBlock(t, blockSSZ) default: t.Fatalf("unknown fork version: %v", fork) } execMock := &engineMock{ powBlocks: make(map[[32]byte]*ethpb.PowBlock), } service := startChainService(t, beaconState, beaconBlock, execMock) var lastTick int64 for _, step := range steps { if step.Tick != nil { newTick := int64(*step.Tick) service.SetGenesisTime(time.Unix(time.Now().Unix()-newTick, 0)) if newTick > lastTick { slot := uint64(newTick) / params.BeaconConfig().SecondsPerSlot require.NoError(t, service.NewSlot(ctx, types.Slot(slot))) lastTick = newTick } } if step.Block != nil { blockFile, err := util.BazelFileBytes(testsFolderPath, folder.Name(), fmt.Sprint(*step.Block, ".ssz_snappy")) require.NoError(t, err) blockSSZ, err := snappy.Decode(nil /* dst */, blockFile) require.NoError(t, err) var beaconBlock interfaces.SignedBeaconBlock switch fork { case version.Phase0: beaconBlock = unmarshalSignedPhase0Block(t, blockSSZ) case version.Altair: beaconBlock = unmarshalSignedAltairBlock(t, blockSSZ) case version.Bellatrix: beaconBlock = unmarshalSignedBellatrixBlock(t, blockSSZ) default: t.Fatalf("unknown fork version: %v", fork) } r, err := beaconBlock.Block().HashTreeRoot() require.NoError(t, err) slotsSinceGenesis := slots.SinceGenesis(service.GenesisTime()) args := &forkchoicetypes.ProposerBoostRootArgs{ BlockRoot: r, BlockSlot: beaconBlock.Block().Slot(), CurrentSlot: slotsSinceGenesis, SecondsIntoSlot: uint64(lastTick) % params.BeaconConfig().SecondsPerSlot, } require.NoError(t, service.ForkChoicer().BoostProposerRoot(ctx, args)) if step.Valid != nil && !*step.Valid { require.Equal(t, true, service.ReceiveBlock(ctx, beaconBlock, r) != nil) } else { require.NoError(t, service.ReceiveBlock(ctx, beaconBlock, r)) } } if step.Attestation != nil { attFile, err := util.BazelFileBytes(testsFolderPath, folder.Name(), fmt.Sprint(*step.Attestation, ".ssz_snappy")) require.NoError(t, err) attSSZ, err := snappy.Decode(nil /* dst */, attFile) require.NoError(t, err) att := ðpb.Attestation{} require.NoError(t, att.UnmarshalSSZ(attSSZ), "Failed to unmarshal") require.NoError(t, service.OnAttestation(ctx, att)) } if step.PowBlock != nil { powBlockFile, err := util.BazelFileBytes(testsFolderPath, folder.Name(), fmt.Sprint(*step.PowBlock, ".ssz_snappy")) require.NoError(t, err) p, err := snappy.Decode(nil /* dst */, powBlockFile) require.NoError(t, err) pb := ðpb.PowBlock{} require.NoError(t, pb.UnmarshalSSZ(p), "Failed to unmarshal") execMock.powBlocks[bytesutil.ToBytes32(pb.BlockHash)] = pb tdInBigEndian := bytesutil.ReverseByteOrder(pb.TotalDifficulty) tdBigint := new(big.Int) tdBigint.SetBytes(tdInBigEndian) } if step.Check != nil { require.NoError(t, service.UpdateAndSaveHeadWithBalances(ctx)) c := step.Check if c.Head != nil { r, err := service.HeadRoot(ctx) require.NoError(t, err) require.DeepEqual(t, common.FromHex(c.Head.Root), r) require.Equal(t, types.Slot(c.Head.Slot), service.HeadSlot()) } if c.JustifiedCheckPoint != nil { cp := ðpb.Checkpoint{ Epoch: types.Epoch(c.JustifiedCheckPoint.Epoch), Root: common.FromHex(c.JustifiedCheckPoint.Root), } require.DeepEqual(t, cp, service.CurrentJustifiedCheckpt()) } if c.BestJustifiedCheckPoint != nil { cp := ðpb.Checkpoint{ Epoch: types.Epoch(c.BestJustifiedCheckPoint.Epoch), Root: common.FromHex(c.BestJustifiedCheckPoint.Root), } require.DeepEqual(t, cp, service.BestJustifiedCheckpt()) } if c.FinalizedCheckPoint != nil { cp := ðpb.Checkpoint{ Epoch: types.Epoch(c.FinalizedCheckPoint.Epoch), Root: common.FromHex(c.FinalizedCheckPoint.Root), } require.DeepSSZEqual(t, cp, service.FinalizedCheckpt()) } if c.ProposerBoostRoot != nil { want := fmt.Sprintf("%#x", common.FromHex(*c.ProposerBoostRoot)) got := fmt.Sprintf("%#x", service.ForkChoiceStore().ProposerBoost()) require.DeepEqual(t, want, got) } } } }) } } } func unmarshalPhase0State(t *testing.T, raw []byte) state.BeaconState { base := ðpb.BeaconState{} require.NoError(t, base.UnmarshalSSZ(raw)) st, err := v1.InitializeFromProto(base) require.NoError(t, err) return st } func unmarshalPhase0Block(t *testing.T, raw []byte) interfaces.SignedBeaconBlock { base := ðpb.BeaconBlock{} require.NoError(t, base.UnmarshalSSZ(raw)) blk, err := wrapper.WrappedSignedBeaconBlock(ðpb.SignedBeaconBlock{Block: base, Signature: make([]byte, fieldparams.BLSSignatureLength)}) require.NoError(t, err) return blk } func unmarshalSignedPhase0Block(t *testing.T, raw []byte) interfaces.SignedBeaconBlock { base := ðpb.SignedBeaconBlock{} require.NoError(t, base.UnmarshalSSZ(raw)) blk, err := wrapper.WrappedSignedBeaconBlock(base) require.NoError(t, err) return blk } func unmarshalAltairState(t *testing.T, raw []byte) state.BeaconState { base := ðpb.BeaconStateAltair{} require.NoError(t, base.UnmarshalSSZ(raw)) st, err := v2.InitializeFromProto(base) require.NoError(t, err) return st } func unmarshalAltairBlock(t *testing.T, raw []byte) interfaces.SignedBeaconBlock { base := ðpb.BeaconBlockAltair{} require.NoError(t, base.UnmarshalSSZ(raw)) blk, err := wrapper.WrappedSignedBeaconBlock(ðpb.SignedBeaconBlockAltair{Block: base, Signature: make([]byte, fieldparams.BLSSignatureLength)}) require.NoError(t, err) return blk } func unmarshalSignedAltairBlock(t *testing.T, raw []byte) interfaces.SignedBeaconBlock { base := ðpb.SignedBeaconBlockAltair{} require.NoError(t, base.UnmarshalSSZ(raw)) blk, err := wrapper.WrappedSignedBeaconBlock(base) require.NoError(t, err) return blk } func unmarshalBellatrixState(t *testing.T, raw []byte) state.BeaconState { base := ðpb.BeaconStateBellatrix{} require.NoError(t, base.UnmarshalSSZ(raw)) st, err := v3.InitializeFromProto(base) require.NoError(t, err) return st } func unmarshalBellatrixBlock(t *testing.T, raw []byte) interfaces.SignedBeaconBlock { base := ðpb.BeaconBlockBellatrix{} require.NoError(t, base.UnmarshalSSZ(raw)) blk, err := wrapper.WrappedSignedBeaconBlock(ðpb.SignedBeaconBlockBellatrix{Block: base, Signature: make([]byte, fieldparams.BLSSignatureLength)}) require.NoError(t, err) return blk } func unmarshalSignedBellatrixBlock(t *testing.T, raw []byte) interfaces.SignedBeaconBlock { base := ðpb.SignedBeaconBlockBellatrix{} require.NoError(t, base.UnmarshalSSZ(raw)) blk, err := wrapper.WrappedSignedBeaconBlock(base) require.NoError(t, err) return blk }