package db import ( "bytes" "reflect" "sort" "testing" "github.com/gogo/protobuf/proto" pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" "github.com/prysmaticlabs/prysm/shared/hashutil" ) func TestSaveAndRetrieveAttestation(t *testing.T) { db := setupDB(t) defer teardownDB(t, db) a := &pb.Attestation{ Data: &pb.AttestationData{ Slot: 0, Shard: 0, }, } if err := db.SaveAttestation(a); err != nil { t.Fatalf("Failed to save attestation: %v", err) } aHash, err := hashutil.HashProto(a) if err != nil { t.Fatalf("Failed to hash Attestation: %v", err) } aPrime, err := db.Attestation(aHash) if err != nil { t.Fatalf("Failed to call Attestation: %v", err) } aEnc, err := proto.Marshal(a) if err != nil { t.Fatalf("Failed to encode: %v", err) } aPrimeEnc, err := proto.Marshal(aPrime) if err != nil { t.Fatalf("Failed to encode: %v", err) } if !bytes.Equal(aEnc, aPrimeEnc) { t.Fatalf("Saved attestation and retrieved attestation are not equal: %#x and %#x", aEnc, aPrimeEnc) } } func TestRetrieveAttestations(t *testing.T) { db := setupDB(t) defer teardownDB(t, db) // Generate 100 unique attestations to save in DB. attestations := make([]*pb.Attestation, 100) for i := 0; i < len(attestations); i++ { attestations[i] = &pb.Attestation{ Data: &pb.AttestationData{ Slot: uint64(i), Shard: uint64(i), }, } if err := db.SaveAttestation(attestations[i]); err != nil { t.Fatalf("Failed to save attestation: %v", err) } } retrievedAttestations, err := db.Attestations() if err != nil { t.Fatalf("Could not retrieve attestations: %v", err) } // Sort the retrieved attestations based on slot ordering for comparison. sort.Slice(retrievedAttestations, func(i, j int) bool { return retrievedAttestations[i].Data.Slot < retrievedAttestations[j].Data.Slot }) if !reflect.DeepEqual(retrievedAttestations, attestations) { t.Log("Retrieved attestations did not match generated attestations") } } func TestDeleteAttestation(t *testing.T) { db := setupDB(t) defer teardownDB(t, db) a := &pb.Attestation{ Data: &pb.AttestationData{ Slot: 0, Shard: 0, }, } if err := db.SaveAttestation(a); err != nil { t.Fatalf("Could not save attestation: %v", err) } aHash, err := hashutil.HashProto(a) if err != nil { t.Fatalf("Failed to hash Attestation: %v", err) } aPrime, err := db.Attestation(aHash) if err != nil { t.Fatalf("Could not call Attestation: %v", err) } if !reflect.DeepEqual(aPrime, a) { t.Errorf("Saved attestation and retrieved attestation are not equal") } if err := db.DeleteAttestation(a); err != nil { t.Fatalf("Could not delete attestation: %v", err) } if db.HasAttestation(aHash) { t.Error("Deleted attestation still there") } } func TestNilAttestation(t *testing.T) { db := setupDB(t) defer teardownDB(t, db) nilHash := [32]byte{} a, err := db.Attestation(nilHash) if err != nil { t.Fatalf("Failed to retrieve nilHash: %v", err) } if a != nil { t.Fatal("Expected nilHash to return no attestation") } } func TestHasAttestation(t *testing.T) { db := setupDB(t) defer teardownDB(t, db) a := &pb.Attestation{ Data: &pb.AttestationData{ Slot: 0, Shard: 0, }, } aHash, err := hashutil.HashProto(a) if err != nil { t.Fatalf("Failed to hash Attestation: %v", err) } if db.HasAttestation(aHash) { t.Fatal("Expected HasAttestation to return false") } if err := db.SaveAttestation(a); err != nil { t.Fatalf("Failed to save attestation: %v", err) } if !db.HasAttestation(aHash) { t.Fatal("Expected HasAttestation to return true") } }