package sync import ( "context" "testing" "time" ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1" "github.com/prysmaticlabs/go-ssz" mockChain "github.com/prysmaticlabs/prysm/beacon-chain/blockchain/testing" dbtest "github.com/prysmaticlabs/prysm/beacon-chain/db/testing" p2ptest "github.com/prysmaticlabs/prysm/beacon-chain/p2p/testing" mockSync "github.com/prysmaticlabs/prysm/beacon-chain/sync/initial-sync/testing" "github.com/prysmaticlabs/prysm/shared/testutil" ) func TestValidateBeaconAttestation_ValidBlock(t *testing.T) { db := dbtest.SetupDB(t) defer dbtest.TeardownDB(t, db) p := p2ptest.NewTestP2P(t) ctx := context.Background() rs := &RegularSync{ db: db, chain: &mockChain.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{ Epoch: 0, }, }, initialSync: &mockSync.Sync{IsSyncing: false}, } blk := ðpb.BeaconBlock{ Slot: 55, } if err := db.SaveBlock(ctx, blk); err != nil { t.Fatal(err) } blockRoot, err := ssz.SigningRoot(blk) if err != nil { t.Fatal(err) } msg := ðpb.Attestation{ Data: ðpb.AttestationData{ BeaconBlockRoot: blockRoot[:], Source: ðpb.Checkpoint{ Epoch: 1, }, Target: ðpb.Checkpoint{ Epoch: 2, }, }, } valid, err := rs.validateBeaconAttestation(ctx, msg, p, false /*fromSelf*/) if err != nil { t.Errorf("Beacon attestation failed validation: %v", err) } if !valid { t.Error("Beacon attestation failed validation") } if !p.BroadcastCalled { t.Error("No message was broadcast") } time.Sleep(100 * time.Millisecond) // It should ignore duplicate identical attestations. p.BroadcastCalled = false valid, _ = rs.validateBeaconAttestation(ctx, msg, p, false /*fromSelf*/) if valid { t.Error("Second identical beacon attestation passed validation when it should not have") } if p.BroadcastCalled { t.Error("Second identcial beacon attestation was re-broadcast") } } func TestValidateBeaconAttestation_InvalidBlock(t *testing.T) { db := dbtest.SetupDB(t) defer dbtest.TeardownDB(t, db) p := p2ptest.NewTestP2P(t) ctx := context.Background() rs := &RegularSync{ db: db, initialSync: &mockSync.Sync{IsSyncing: false}, } msg := ðpb.Attestation{ Data: ðpb.AttestationData{ BeaconBlockRoot: testutil.Random32Bytes(t), }, } valid, _ := rs.validateBeaconAttestation(ctx, msg, p, false /*fromSelf*/) if valid { t.Error("Invalid beacon attestation passed validation when it should not have") } if p.BroadcastCalled { t.Error("Invalid beacon attestation was broadcast") } } func TestValidateBeaconAttestation_ValidBlock_FromSelf(t *testing.T) { db := dbtest.SetupDB(t) defer dbtest.TeardownDB(t, db) p := p2ptest.NewTestP2P(t) ctx := context.Background() rs := &RegularSync{ db: db, initialSync: &mockSync.Sync{IsSyncing: false}, } blk := ðpb.BeaconBlock{ Slot: 55, } if err := db.SaveBlock(ctx, blk); err != nil { t.Fatal(err) } blockRoot, err := ssz.SigningRoot(blk) if err != nil { t.Fatal(err) } msg := ðpb.Attestation{ Data: ðpb.AttestationData{ BeaconBlockRoot: blockRoot[:], }, } valid, _ := rs.validateBeaconAttestation(ctx, msg, p, true /*fromSelf*/) if valid { t.Error("Beacon attestation passed validation") } if p.BroadcastCalled { t.Error("Message was broadcast") } } func TestValidateBeaconAttestation_Syncing(t *testing.T) { db := dbtest.SetupDB(t) defer dbtest.TeardownDB(t, db) p := p2ptest.NewTestP2P(t) ctx := context.Background() rs := &RegularSync{ db: db, initialSync: &mockSync.Sync{IsSyncing: true}, } blk := ðpb.BeaconBlock{ Slot: 55, } if err := db.SaveBlock(ctx, blk); err != nil { t.Fatal(err) } blockRoot, err := ssz.SigningRoot(blk) if err != nil { t.Fatal(err) } msg := ðpb.Attestation{ Data: ðpb.AttestationData{ BeaconBlockRoot: blockRoot[:], }, } valid, err := rs.validateBeaconAttestation(ctx, msg, p, false /*fromSelf*/) if valid { t.Error("Beacon attestation passed validation") } } func TestValidateBeaconAttestation_OldAttestation(t *testing.T) { db := dbtest.SetupDB(t) defer dbtest.TeardownDB(t, db) p := p2ptest.NewTestP2P(t) ctx := context.Background() rs := &RegularSync{ db: db, chain: &mockChain.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{ Epoch: 10, }, }, initialSync: &mockSync.Sync{IsSyncing: false}, } blk := ðpb.BeaconBlock{ Slot: 55, } if err := db.SaveBlock(ctx, blk); err != nil { t.Fatal(err) } blockRoot, err := ssz.SigningRoot(blk) if err != nil { t.Fatal(err) } msg := ðpb.Attestation{ Data: ðpb.AttestationData{ BeaconBlockRoot: blockRoot[:], Source: ðpb.Checkpoint{ Epoch: 10, }, Target: ðpb.Checkpoint{ Epoch: 2, }, }, } valid, err := rs.validateBeaconAttestation(ctx, msg, p, false /*fromSelf*/) if err != nil { t.Errorf("Beacon attestation failed validation: %v", err) } if valid { t.Error("Beacon attestation passed validation when it should have failed") } if p.BroadcastCalled { t.Error("Message was broadcasted") } // source and target epoch same as finalized checkpoint msg = ðpb.Attestation{ Data: ðpb.AttestationData{ BeaconBlockRoot: blockRoot[:], Source: ðpb.Checkpoint{ Epoch: 10, }, Target: ðpb.Checkpoint{ Epoch: 10, }, }, } valid, err = rs.validateBeaconAttestation(ctx, msg, p, false /*fromSelf*/) if err != nil { t.Errorf("Beacon attestation failed validation: %v", err) } if valid { t.Error("Beacon attestation passed validation when it should have failed") } if p.BroadcastCalled { t.Error("Message was broadcasted") } } func TestValidateBeaconAttestation_FirstEpoch(t *testing.T) { db := dbtest.SetupDB(t) defer dbtest.TeardownDB(t, db) p := p2ptest.NewTestP2P(t) ctx := context.Background() rs := &RegularSync{ db: db, chain: &mockChain.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{ Epoch: 0, }, }, initialSync: &mockSync.Sync{IsSyncing: false}, } blk := ðpb.BeaconBlock{ Slot: 1, } if err := db.SaveBlock(ctx, blk); err != nil { t.Fatal(err) } blockRoot, err := ssz.SigningRoot(blk) if err != nil { t.Fatal(err) } // Attestation at genesis epoch should not be rejected msg := ðpb.Attestation{ Data: ðpb.AttestationData{ BeaconBlockRoot: blockRoot[:], Source: ðpb.Checkpoint{ Epoch: 0, }, Target: ðpb.Checkpoint{ Epoch: 0, }, }, } valid, err := rs.validateBeaconAttestation(ctx, msg, p, false /*fromSelf*/) if err != nil { t.Errorf("Beacon attestation failed validation: %v", err) } if !valid { t.Error("Beacon attestation did not pass validation") } if !p.BroadcastCalled { t.Error("Message was not broadcasted") } }