2019-01-11 10:55:01 +00:00
|
|
|
package backend
|
|
|
|
|
|
|
|
import (
|
2019-02-20 05:07:28 +00:00
|
|
|
"crypto/rand"
|
|
|
|
"encoding/binary"
|
2019-01-11 10:55:01 +00:00
|
|
|
"fmt"
|
2019-01-12 01:10:39 +00:00
|
|
|
"time"
|
|
|
|
|
2019-02-19 18:05:34 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
2019-01-11 10:55:01 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2019-02-20 05:07:28 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/bls"
|
2019-02-20 18:58:34 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/forkutils"
|
2019-01-11 10:55:01 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2019-02-20 05:07:28 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/ssz"
|
2019-01-31 02:53:58 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/trieutil"
|
2019-01-11 10:55:01 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// 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,
|
2019-01-31 02:53:58 +00:00
|
|
|
depositsTrie *trieutil.DepositTrie,
|
2019-02-03 10:58:36 +00:00
|
|
|
simObjects *SimulatedObjects,
|
2019-02-20 05:07:28 +00:00
|
|
|
privKeys []*bls.SecretKey,
|
2019-01-11 10:55:01 +00:00
|
|
|
) (*pb.BeaconBlock, [32]byte, error) {
|
2019-02-14 20:04:47 +00:00
|
|
|
stateRoot, err := ssz.TreeHash(beaconState)
|
2019-01-11 10:55:01 +00:00
|
|
|
if err != nil {
|
2019-02-14 20:04:47 +00:00
|
|
|
return nil, [32]byte{}, fmt.Errorf("could not tree hash state: %v", err)
|
2019-01-11 10:55:01 +00:00
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
proposerIdx, err := helpers.BeaconProposerIndex(beaconState, beaconState.Slot+1)
|
|
|
|
if err != nil {
|
|
|
|
return nil, [32]byte{}, err
|
|
|
|
}
|
|
|
|
epoch := helpers.SlotToEpoch(beaconState.Slot + 1)
|
|
|
|
buf := make([]byte, 32)
|
|
|
|
binary.LittleEndian.PutUint64(buf, epoch)
|
2019-02-20 18:58:34 +00:00
|
|
|
domain := forkutils.DomainVersion(beaconState.Fork, epoch, params.BeaconConfig().DomainRandao)
|
2019-02-20 05:07:28 +00:00
|
|
|
// We make the previous validator's index sign the message instead of the proposer.
|
|
|
|
epochSignature := privKeys[proposerIdx].Sign(buf, domain)
|
2019-01-11 10:55:01 +00:00
|
|
|
block := &pb.BeaconBlock{
|
2019-02-19 20:24:00 +00:00
|
|
|
Slot: beaconState.Slot + 1,
|
2019-02-20 05:07:28 +00:00
|
|
|
RandaoReveal: epochSignature.Marshal(),
|
2019-02-19 20:24:00 +00:00
|
|
|
ParentRootHash32: prevBlockRoot[:],
|
|
|
|
StateRootHash32: stateRoot[:],
|
2019-02-02 21:24:42 +00:00
|
|
|
Eth1Data: &pb.Eth1Data{
|
|
|
|
DepositRootHash32: []byte{1},
|
|
|
|
BlockHash32: []byte{2},
|
|
|
|
},
|
2019-01-11 10:55:01 +00:00
|
|
|
Body: &pb.BeaconBlockBody{
|
|
|
|
ProposerSlashings: []*pb.ProposerSlashing{},
|
2019-01-30 10:11:13 +00:00
|
|
|
AttesterSlashings: []*pb.AttesterSlashing{},
|
2019-01-11 10:55:01 +00:00
|
|
|
Attestations: []*pb.Attestation{},
|
|
|
|
Deposits: []*pb.Deposit{},
|
2019-02-18 16:52:16 +00:00
|
|
|
VoluntaryExits: []*pb.VoluntaryExit{},
|
2019-01-11 10:55:01 +00:00
|
|
|
},
|
|
|
|
}
|
2019-02-03 10:58:36 +00:00
|
|
|
if simObjects.simDeposit != nil {
|
2019-01-12 01:10:39 +00:00
|
|
|
depositInput := &pb.DepositInput{
|
2019-02-03 10:58:36 +00:00
|
|
|
Pubkey: []byte(simObjects.simDeposit.Pubkey),
|
2019-02-09 20:35:51 +00:00
|
|
|
WithdrawalCredentialsHash32: make([]byte, 32),
|
|
|
|
ProofOfPossession: make([]byte, 96),
|
2019-01-12 01:10:39 +00:00
|
|
|
}
|
|
|
|
|
2019-02-19 18:05:34 +00:00
|
|
|
data, err := helpers.EncodeDepositData(depositInput, simObjects.simDeposit.Amount, time.Now().Unix())
|
2019-01-12 01:10:39 +00:00
|
|
|
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)
|
2019-02-08 17:01:15 +00:00
|
|
|
merkleBranch := depositsTrie.Branch()
|
2019-01-12 01:10:39 +00:00
|
|
|
|
|
|
|
block.Body.Deposits = append(block.Body.Deposits, &pb.Deposit{
|
|
|
|
DepositData: data,
|
|
|
|
MerkleBranchHash32S: merkleBranch,
|
2019-02-03 10:58:36 +00:00
|
|
|
MerkleTreeIndex: simObjects.simDeposit.MerkleIndex,
|
2019-01-12 01:10:39 +00:00
|
|
|
})
|
|
|
|
}
|
2019-02-03 10:58:36 +00:00
|
|
|
if simObjects.simProposerSlashing != nil {
|
2019-01-12 02:11:43 +00:00
|
|
|
block.Body.ProposerSlashings = append(block.Body.ProposerSlashings, &pb.ProposerSlashing{
|
2019-02-03 10:58:36 +00:00
|
|
|
ProposerIndex: simObjects.simProposerSlashing.ProposerIndex,
|
2019-01-12 02:11:43 +00:00
|
|
|
ProposalData_1: &pb.ProposalSignedData{
|
2019-02-03 10:58:36 +00:00
|
|
|
Slot: simObjects.simProposerSlashing.Proposal1Slot,
|
|
|
|
Shard: simObjects.simProposerSlashing.Proposal1Shard,
|
|
|
|
BlockRootHash32: []byte(simObjects.simProposerSlashing.Proposal1Root),
|
2019-01-12 02:11:43 +00:00
|
|
|
},
|
|
|
|
ProposalData_2: &pb.ProposalSignedData{
|
2019-02-03 10:58:36 +00:00
|
|
|
Slot: simObjects.simProposerSlashing.Proposal2Slot,
|
|
|
|
Shard: simObjects.simProposerSlashing.Proposal2Shard,
|
|
|
|
BlockRootHash32: []byte(simObjects.simProposerSlashing.Proposal2Root),
|
2019-01-12 02:11:43 +00:00
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
2019-02-03 10:58:36 +00:00
|
|
|
if simObjects.simAttesterSlashing != nil {
|
2019-01-30 10:11:13 +00:00
|
|
|
block.Body.AttesterSlashings = append(block.Body.AttesterSlashings, &pb.AttesterSlashing{
|
2019-02-08 21:53:36 +00:00
|
|
|
SlashableAttestation_1: &pb.SlashableAttestation{
|
2019-01-14 15:54:27 +00:00
|
|
|
Data: &pb.AttestationData{
|
2019-02-08 21:53:36 +00:00
|
|
|
Slot: simObjects.simAttesterSlashing.SlashableAttestation1Slot,
|
|
|
|
JustifiedEpoch: simObjects.simAttesterSlashing.SlashableAttestation1JustifiedEpoch,
|
2019-01-14 15:54:27 +00:00
|
|
|
},
|
2019-02-08 21:53:36 +00:00
|
|
|
CustodyBitfield: []byte(simObjects.simAttesterSlashing.SlashableAttestation1CustodyBitField),
|
|
|
|
ValidatorIndices: simObjects.simAttesterSlashing.SlashableAttestation1ValidatorIndices,
|
2019-01-14 15:54:27 +00:00
|
|
|
},
|
2019-02-08 21:53:36 +00:00
|
|
|
SlashableAttestation_2: &pb.SlashableAttestation{
|
2019-01-14 15:54:27 +00:00
|
|
|
Data: &pb.AttestationData{
|
2019-02-08 21:53:36 +00:00
|
|
|
Slot: simObjects.simAttesterSlashing.SlashableAttestation2Slot,
|
|
|
|
JustifiedEpoch: simObjects.simAttesterSlashing.SlashableAttestation2JustifiedEpoch,
|
2019-01-14 15:54:27 +00:00
|
|
|
},
|
2019-02-08 21:53:36 +00:00
|
|
|
CustodyBitfield: []byte(simObjects.simAttesterSlashing.SlashableAttestation2CustodyBitField),
|
|
|
|
ValidatorIndices: simObjects.simAttesterSlashing.SlashableAttestation2ValidatorIndices,
|
2019-01-14 15:54:27 +00:00
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
2019-02-03 10:58:36 +00:00
|
|
|
if simObjects.simValidatorExit != nil {
|
2019-02-18 16:52:16 +00:00
|
|
|
block.Body.VoluntaryExits = append(block.Body.VoluntaryExits, &pb.VoluntaryExit{
|
2019-02-08 21:53:36 +00:00
|
|
|
Epoch: simObjects.simValidatorExit.Epoch,
|
2019-02-03 10:58:36 +00:00
|
|
|
ValidatorIndex: simObjects.simValidatorExit.ValidatorIndex,
|
2019-01-14 17:02:49 +00:00
|
|
|
})
|
|
|
|
}
|
2019-02-14 20:04:47 +00:00
|
|
|
blockRoot, err := ssz.TreeHash(block)
|
2019-01-11 10:55:01 +00:00
|
|
|
if err != nil {
|
2019-02-14 20:04:47 +00:00
|
|
|
return nil, [32]byte{}, fmt.Errorf("could not tree hash new block: %v", err)
|
2019-01-11 10:55:01 +00:00
|
|
|
}
|
2019-02-14 20:04:47 +00:00
|
|
|
return block, blockRoot, nil
|
2019-01-11 10:55:01 +00:00
|
|
|
}
|
|
|
|
|
2019-02-23 06:06:20 +00:00
|
|
|
// generateInitialSimulatedDeposits generates initial deposits for creating a beacon state in the simulated
|
2019-01-11 10:55:01 +00:00
|
|
|
// backend based on the yaml configuration.
|
2019-02-20 05:07:28 +00:00
|
|
|
func generateInitialSimulatedDeposits(numDeposits uint64) ([]*pb.Deposit, []*bls.SecretKey, error) {
|
2019-02-14 23:01:07 +00:00
|
|
|
genesisTime := time.Date(2018, 9, 0, 0, 0, 0, 0, time.UTC).Unix()
|
2019-02-09 20:35:51 +00:00
|
|
|
deposits := make([]*pb.Deposit, numDeposits)
|
2019-02-20 05:07:28 +00:00
|
|
|
privKeys := make([]*bls.SecretKey, numDeposits)
|
2019-01-11 10:55:01 +00:00
|
|
|
for i := 0; i < len(deposits); i++ {
|
2019-02-20 05:07:28 +00:00
|
|
|
priv, err := bls.RandKey(rand.Reader)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, fmt.Errorf("could not initialize key: %v", err)
|
|
|
|
}
|
2019-01-11 10:55:01 +00:00
|
|
|
depositInput := &pb.DepositInput{
|
2019-02-20 05:07:28 +00:00
|
|
|
Pubkey: priv.PublicKey().Marshal(),
|
2019-02-09 20:35:51 +00:00
|
|
|
WithdrawalCredentialsHash32: make([]byte, 32),
|
|
|
|
ProofOfPossession: make([]byte, 96),
|
2019-01-11 10:55:01 +00:00
|
|
|
}
|
2019-02-19 18:05:34 +00:00
|
|
|
depositData, err := helpers.EncodeDepositData(
|
2019-01-11 10:55:01 +00:00
|
|
|
depositInput,
|
2019-02-09 13:09:09 +00:00
|
|
|
params.BeaconConfig().MaxDepositAmount,
|
2019-01-11 10:55:01 +00:00
|
|
|
genesisTime,
|
|
|
|
)
|
|
|
|
if err != nil {
|
2019-02-20 05:07:28 +00:00
|
|
|
return nil, nil, fmt.Errorf("could not encode genesis block deposits: %v", err)
|
2019-01-11 10:55:01 +00:00
|
|
|
}
|
|
|
|
deposits[i] = &pb.Deposit{DepositData: depositData}
|
2019-02-20 05:07:28 +00:00
|
|
|
privKeys[i] = priv
|
2019-01-11 10:55:01 +00:00
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
return deposits, privKeys, nil
|
2019-01-11 10:55:01 +00:00
|
|
|
}
|