package backend import ( "fmt" "strconv" "time" "github.com/prysmaticlabs/prysm/shared/trie" "github.com/gogo/protobuf/proto" b "github.com/prysmaticlabs/prysm/beacon-chain/core/blocks" pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" "github.com/prysmaticlabs/prysm/shared/hashutil" "github.com/prysmaticlabs/prysm/shared/params" ) // Generates a simulated beacon block to use // in the next state transition given the current state, // the previous beacon block, and previous beacon block root. func generateSimulatedBlock( beaconState *pb.BeaconState, prevBlockRoot [32]byte, randaoReveal [32]byte, depositRandaoCommit [32]byte, simulatedDeposit *StateTestDeposit, depositsTrie *trie.DepositTrie, simulatedProposerSlashing *StateTestProposerSlashing, ) (*pb.BeaconBlock, [32]byte, error) { encodedState, err := proto.Marshal(beaconState) if err != nil { return nil, [32]byte{}, fmt.Errorf("could not marshal beacon state: %v", err) } stateRoot := hashutil.Hash(encodedState) block := &pb.BeaconBlock{ Slot: beaconState.Slot + 1, RandaoRevealHash32: randaoReveal[:], ParentRootHash32: prevBlockRoot[:], StateRootHash32: stateRoot[:], Body: &pb.BeaconBlockBody{ ProposerSlashings: []*pb.ProposerSlashing{}, CasperSlashings: []*pb.CasperSlashing{}, Attestations: []*pb.Attestation{}, Deposits: []*pb.Deposit{}, Exits: []*pb.Exit{}, }, } if simulatedDeposit != nil { depositInput := &pb.DepositInput{ Pubkey: []byte(simulatedDeposit.Pubkey), WithdrawalCredentialsHash32: []byte{}, ProofOfPossession: []byte{}, RandaoCommitmentHash32: depositRandaoCommit[:], CustodyCommitmentHash32: []byte{}, } data, err := b.EncodeDepositData(depositInput, simulatedDeposit.Amount, time.Now().Unix()) if err != nil { return nil, [32]byte{}, fmt.Errorf("could not encode deposit data: %v", err) } // We then update the deposits Merkle trie with the deposit data and return // its Merkle branch leading up to the root of the trie. depositsTrie.UpdateDepositTrie(data) merkleBranch := depositsTrie.GenerateMerkleBranch(simulatedDeposit.MerkleIndex) block.Body.Deposits = append(block.Body.Deposits, &pb.Deposit{ DepositData: data, MerkleBranchHash32S: merkleBranch, MerkleTreeIndex: simulatedDeposit.MerkleIndex, }) } if simulatedProposerSlashing != nil { block.Body.ProposerSlashings = append(block.Body.ProposerSlashings, &pb.ProposerSlashing{ ProposerIndex: simulatedProposerSlashing.ProposerIndex, ProposalData_1: &pb.ProposalSignedData{ Slot: simulatedProposerSlashing.Proposal1Slot, Shard: simulatedProposerSlashing.Proposal1Shard, BlockRootHash32: []byte(simulatedProposerSlashing.Proposal1Root), }, ProposalData_2: &pb.ProposalSignedData{ Slot: simulatedProposerSlashing.Proposal2Slot, Shard: simulatedProposerSlashing.Proposal2Shard, BlockRootHash32: []byte(simulatedProposerSlashing.Proposal2Root), }, }) } encodedBlock, err := proto.Marshal(block) if err != nil { return nil, [32]byte{}, fmt.Errorf("could not marshal new block: %v", err) } return block, hashutil.Hash(encodedBlock), nil } // Given a number of slots, we create a list of hash onions from an underlying randao reveal. For example, // if we have N slots, we create a list of [secret, hash(secret), hash(hash(secret)), hash(...(prev N-1 hashes))]. func generateSimulatedRandaoHashOnions(numSlots uint64) [][32]byte { // We create a list of randao hash onions for the given number of epochs // we run the state transition. numEpochs := numSlots % params.BeaconConfig().EpochLength hashOnions := [][32]byte{params.BeaconConfig().SimulatedBlockRandao} // We make the length of the hash onions list equal to the number of epochs + 10 to be safe. for i := uint64(0); i < numEpochs+10; i++ { prevHash := hashOnions[i] hashOnions = append(hashOnions, hashutil.Hash(prevHash[:])) } return hashOnions } // This function determines the block randao reveal assuming there are no skipped slots, // given a list of randao hash onions such as [pre-image, 0x01, 0x02, 0x03], for the // 0th epoch, the block randao reveal will be 0x02 and the proposer commitment 0x03. // The next epoch, the block randao reveal will be 0x01 and the commitment 0x02, // so on and so forth until all randao layers are peeled off. func determineSimulatedBlockRandaoReveal(layersPeeled int, hashOnions [][32]byte) [32]byte { if layersPeeled == 0 { return hashOnions[len(hashOnions)-2] } return hashOnions[len(hashOnions)-layersPeeled-2] } // Generates initial deposits for creating a beacon state in the simulated // backend based on the yaml configuration. func generateInitialSimulatedDeposits(randaoCommit [32]byte) ([]*pb.Deposit, error) { genesisTime := params.BeaconConfig().GenesisTime.Unix() deposits := make([]*pb.Deposit, params.BeaconConfig().DepositsForChainStart) for i := 0; i < len(deposits); i++ { depositInput := &pb.DepositInput{ Pubkey: []byte(strconv.Itoa(i)), RandaoCommitmentHash32: randaoCommit[:], } depositData, err := b.EncodeDepositData( depositInput, params.BeaconConfig().MaxDepositInGwei, genesisTime, ) if err != nil { return nil, fmt.Errorf("could not encode initial block deposits: %v", err) } deposits[i] = &pb.Deposit{DepositData: depositData} } return deposits, nil } // Finds the index of the next slot's proposer in the beacon state's // validator set. func findNextSlotProposerIndex(beaconState *pb.BeaconState) (uint32, error) { nextSlot := beaconState.Slot + 1 epochLength := params.BeaconConfig().EpochLength var earliestSlot uint64 // If the state slot is less than epochLength, then the earliestSlot would // result in a negative number. Therefore we should default to // earliestSlot = 0 in this case. if nextSlot > epochLength { earliestSlot = nextSlot - (nextSlot % epochLength) - epochLength } if nextSlot < earliestSlot || nextSlot >= earliestSlot+(epochLength*2) { return 0, fmt.Errorf("slot %d out of bounds: %d <= slot < %d", nextSlot, earliestSlot, earliestSlot+(epochLength*2), ) } committeeArray := beaconState.ShardCommitteesAtSlots[nextSlot-earliestSlot] firstCommittee := committeeArray.ArrayShardCommittee[0].Committee return firstCommittee[nextSlot%uint64(len(firstCommittee))], nil }