2019-01-19 02:57:51 +00:00
|
|
|
package blockchain
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
b "github.com/prysmaticlabs/prysm/beacon-chain/core/blocks"
|
2019-02-04 20:34:33 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
2019-01-19 02:57:51 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/state"
|
2019-01-21 05:35:34 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/db"
|
2019-01-19 02:57:51 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/internal"
|
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2019-03-03 17:31:29 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
2019-01-19 02:57:51 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Generates an initial genesis block and state using a custom number of initial
|
|
|
|
// deposits as a helper function for LMD Ghost fork-choice testing.
|
|
|
|
func generateTestGenesisStateAndBlock(
|
|
|
|
t testing.TB,
|
|
|
|
numDeposits uint64,
|
|
|
|
beaconDB *db.BeaconDB,
|
|
|
|
) (*pb.BeaconState, *pb.BeaconBlock, [32]byte, [32]byte) {
|
|
|
|
deposits := make([]*pb.Deposit, numDeposits)
|
|
|
|
for i := 0; i < len(deposits); i++ {
|
|
|
|
pubkey := []byte{byte(i)}
|
|
|
|
depositInput := &pb.DepositInput{
|
|
|
|
Pubkey: pubkey,
|
|
|
|
}
|
2019-03-07 03:02:47 +00:00
|
|
|
|
2019-02-09 13:09:09 +00:00
|
|
|
balance := params.BeaconConfig().MaxDepositAmount
|
2019-02-19 18:05:34 +00:00
|
|
|
depositData, err := helpers.EncodeDepositData(depositInput, balance, time.Now().Unix())
|
2019-01-19 02:57:51 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not encode deposit: %v", err)
|
|
|
|
}
|
|
|
|
deposits[i] = &pb.Deposit{DepositData: depositData}
|
|
|
|
}
|
2019-02-14 23:01:07 +00:00
|
|
|
genesisTime := uint64(time.Unix(0, 0).Unix())
|
2019-02-18 16:52:16 +00:00
|
|
|
beaconState, err := state.GenesisBeaconState(deposits, genesisTime, nil)
|
2019-01-19 02:57:51 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2019-02-14 20:04:47 +00:00
|
|
|
if err := beaconDB.SaveState(beaconState); err != nil {
|
2019-01-19 02:57:51 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-02-26 03:42:31 +00:00
|
|
|
stateRoot, err := hashutil.HashProto(beaconState)
|
2019-02-14 20:04:47 +00:00
|
|
|
if err != nil {
|
2019-01-19 02:57:51 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-02-14 20:04:47 +00:00
|
|
|
genesisBlock := b.NewGenesisBlock(stateRoot[:])
|
2019-01-19 02:57:51 +00:00
|
|
|
if err := beaconDB.SaveBlock(genesisBlock); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-02-26 03:42:31 +00:00
|
|
|
genesisRoot, err := hashutil.HashBeaconBlock(genesisBlock)
|
2019-01-19 02:57:51 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-02-14 20:04:47 +00:00
|
|
|
return beaconState, genesisBlock, stateRoot, genesisRoot
|
2019-01-19 02:57:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func setupConflictingBlocks(
|
|
|
|
t *testing.T,
|
|
|
|
beaconDB *db.BeaconDB,
|
|
|
|
genesisHash [32]byte,
|
|
|
|
stateRoot [32]byte,
|
|
|
|
) (candidate1 *pb.BeaconBlock, candidate2 *pb.BeaconBlock) {
|
|
|
|
candidate1 = &pb.BeaconBlock{
|
|
|
|
Slot: 5,
|
|
|
|
ParentRootHash32: genesisHash[:],
|
|
|
|
StateRootHash32: stateRoot[:],
|
|
|
|
}
|
|
|
|
candidate2 = &pb.BeaconBlock{
|
|
|
|
Slot: 5,
|
|
|
|
ParentRootHash32: genesisHash[:],
|
|
|
|
StateRootHash32: []byte("some-other-state"),
|
|
|
|
}
|
|
|
|
// We store these potential heads in the DB.
|
|
|
|
if err := beaconDB.SaveBlock(candidate1); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := beaconDB.SaveBlock(candidate2); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
return candidate1, candidate2
|
|
|
|
}
|
|
|
|
|
2019-03-08 13:55:55 +00:00
|
|
|
func TestVoteCount_ParentDoesNotExistNoVoteCount(t *testing.T) {
|
2019-01-19 02:57:51 +00:00
|
|
|
beaconDB := internal.SetupDB(t)
|
|
|
|
defer internal.TeardownDB(t, beaconDB)
|
2019-02-14 20:04:47 +00:00
|
|
|
genesisBlock := b.NewGenesisBlock([]byte("stateroot"))
|
2019-01-19 02:57:51 +00:00
|
|
|
if err := beaconDB.SaveBlock(genesisBlock); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
potentialHead := &pb.BeaconBlock{
|
2019-03-08 13:55:55 +00:00
|
|
|
ParentRootHash32: []byte{'A'}, // We give a bogus parent root hash.
|
2019-01-19 02:57:51 +00:00
|
|
|
}
|
|
|
|
if err := beaconDB.SaveBlock(potentialHead); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2019-02-13 23:13:00 +00:00
|
|
|
voteTargets := make(map[uint64]*pb.BeaconBlock)
|
|
|
|
voteTargets[0] = potentialHead
|
2019-03-08 13:55:55 +00:00
|
|
|
count, err := VoteCount(genesisBlock, &pb.BeaconState{}, voteTargets, beaconDB)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not get vote count: %v", err)
|
|
|
|
}
|
|
|
|
if count != 0 {
|
|
|
|
t.Errorf("Wanted vote count 0, got: %d", count)
|
2019-01-19 02:57:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestVoteCount_IncreaseCountCorrectly(t *testing.T) {
|
|
|
|
beaconDB := internal.SetupDB(t)
|
|
|
|
defer internal.TeardownDB(t, beaconDB)
|
2019-02-14 20:04:47 +00:00
|
|
|
genesisBlock := b.NewGenesisBlock([]byte("stateroot"))
|
2019-02-26 03:42:31 +00:00
|
|
|
genesisRoot, err := hashutil.HashBeaconBlock(genesisBlock)
|
2019-01-19 02:57:51 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := beaconDB.SaveBlock(genesisBlock); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
potentialHead := &pb.BeaconBlock{
|
2019-03-08 13:55:55 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 5,
|
2019-02-14 20:04:47 +00:00
|
|
|
ParentRootHash32: genesisRoot[:],
|
2019-01-19 02:57:51 +00:00
|
|
|
}
|
|
|
|
potentialHead2 := &pb.BeaconBlock{
|
2019-03-08 13:55:55 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 6,
|
2019-02-14 20:04:47 +00:00
|
|
|
ParentRootHash32: genesisRoot[:],
|
2019-01-19 02:57:51 +00:00
|
|
|
}
|
|
|
|
// We store these potential heads in the DB.
|
|
|
|
if err := beaconDB.SaveBlock(potentialHead); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := beaconDB.SaveBlock(potentialHead2); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-02-13 23:13:00 +00:00
|
|
|
beaconState := &pb.BeaconState{ValidatorBalances: []uint64{1e9, 1e9}}
|
|
|
|
voteTargets := make(map[uint64]*pb.BeaconBlock)
|
|
|
|
voteTargets[0] = potentialHead
|
|
|
|
voteTargets[1] = potentialHead2
|
|
|
|
count, err := VoteCount(genesisBlock, beaconState, voteTargets, beaconDB)
|
2019-01-19 02:57:51 +00:00
|
|
|
if err != nil {
|
2019-02-13 23:13:00 +00:00
|
|
|
t.Fatalf("Could not fetch vote balances: %v", err)
|
2019-01-19 02:57:51 +00:00
|
|
|
}
|
2019-02-13 23:13:00 +00:00
|
|
|
if count != 2e9 {
|
|
|
|
t.Errorf("Expected total balances 2e9, received %d", count)
|
2019-01-19 02:57:51 +00:00
|
|
|
}
|
|
|
|
}
|