2023-05-09 21:38:13 +00:00
|
|
|
package util
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2024-02-15 05:46:47 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/core/helpers"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/state"
|
|
|
|
state_native "github.com/prysmaticlabs/prysm/v5/beacon-chain/state/state-native"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/state/stateutil"
|
|
|
|
fieldparams "github.com/prysmaticlabs/prysm/v5/config/fieldparams"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/config/params"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/crypto/bls"
|
|
|
|
enginev1 "github.com/prysmaticlabs/prysm/v5/proto/engine/v1"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1"
|
2023-05-09 21:38:13 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// DeterministicGenesisStateDeneb returns a genesis state in Deneb format made using the deterministic deposits.
|
|
|
|
func DeterministicGenesisStateDeneb(t testing.TB, numValidators uint64) (state.BeaconState, []bls.SecretKey) {
|
|
|
|
deposits, privKeys, err := DeterministicDepositsAndKeys(numValidators)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(errors.Wrapf(err, "failed to get %d deposits", numValidators))
|
|
|
|
}
|
|
|
|
eth1Data, err := DeterministicEth1Data(len(deposits))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(errors.Wrapf(err, "failed to get eth1data for %d deposits", numValidators))
|
|
|
|
}
|
|
|
|
beaconState, err := genesisBeaconStateDeneb(context.Background(), deposits, uint64(0), eth1Data)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(errors.Wrapf(err, "failed to get genesis beacon state of %d validators", numValidators))
|
|
|
|
}
|
|
|
|
resetCache()
|
|
|
|
return beaconState, privKeys
|
|
|
|
}
|
|
|
|
|
|
|
|
// genesisBeaconStateDeneb returns the genesis beacon state.
|
|
|
|
func genesisBeaconStateDeneb(ctx context.Context, deposits []*ethpb.Deposit, genesisTime uint64, eth1Data *ethpb.Eth1Data) (state.BeaconState, error) {
|
|
|
|
st, err := emptyGenesisStateDeneb()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Process initial deposits.
|
|
|
|
st, err = helpers.UpdateGenesisEth1Data(st, deposits, eth1Data)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
st, err = processPreGenesisDeposits(ctx, st, deposits)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not process validator deposits")
|
|
|
|
}
|
|
|
|
|
|
|
|
return buildGenesisBeaconStateDeneb(genesisTime, st, st.Eth1Data())
|
|
|
|
}
|
|
|
|
|
|
|
|
// emptyGenesisStateDeneb returns an empty genesis state in Deneb format.
|
|
|
|
func emptyGenesisStateDeneb() (state.BeaconState, error) {
|
|
|
|
st := ðpb.BeaconStateDeneb{
|
|
|
|
// Misc fields.
|
|
|
|
Slot: 0,
|
|
|
|
Fork: ðpb.Fork{
|
|
|
|
PreviousVersion: params.BeaconConfig().BellatrixForkVersion,
|
|
|
|
CurrentVersion: params.BeaconConfig().DenebForkVersion,
|
|
|
|
Epoch: 0,
|
|
|
|
},
|
|
|
|
// Validator registry fields.
|
|
|
|
Validators: []*ethpb.Validator{},
|
|
|
|
Balances: []uint64{},
|
|
|
|
InactivityScores: []uint64{},
|
|
|
|
|
|
|
|
JustificationBits: []byte{0},
|
|
|
|
HistoricalRoots: [][]byte{},
|
|
|
|
CurrentEpochParticipation: []byte{},
|
|
|
|
PreviousEpochParticipation: []byte{},
|
|
|
|
|
|
|
|
// Eth1 data.
|
|
|
|
Eth1Data: ðpb.Eth1Data{},
|
|
|
|
Eth1DataVotes: []*ethpb.Eth1Data{},
|
|
|
|
Eth1DepositIndex: 0,
|
|
|
|
|
|
|
|
LatestExecutionPayloadHeader: &enginev1.ExecutionPayloadHeaderDeneb{},
|
|
|
|
}
|
|
|
|
return state_native.InitializeFromProtoDeneb(st)
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildGenesisBeaconStateDeneb(genesisTime uint64, preState state.BeaconState, eth1Data *ethpb.Eth1Data) (state.BeaconState, error) {
|
|
|
|
if eth1Data == nil {
|
|
|
|
return nil, errors.New("no eth1data provided for genesis state")
|
|
|
|
}
|
|
|
|
|
|
|
|
randaoMixes := make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector)
|
|
|
|
for i := 0; i < len(randaoMixes); i++ {
|
|
|
|
h := make([]byte, 32)
|
|
|
|
copy(h, eth1Data.BlockHash)
|
|
|
|
randaoMixes[i] = h
|
|
|
|
}
|
|
|
|
|
|
|
|
zeroHash := params.BeaconConfig().ZeroHash[:]
|
|
|
|
|
|
|
|
activeIndexRoots := make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector)
|
|
|
|
for i := 0; i < len(activeIndexRoots); i++ {
|
|
|
|
activeIndexRoots[i] = zeroHash
|
|
|
|
}
|
|
|
|
|
|
|
|
blockRoots := make([][]byte, params.BeaconConfig().SlotsPerHistoricalRoot)
|
|
|
|
for i := 0; i < len(blockRoots); i++ {
|
|
|
|
blockRoots[i] = zeroHash
|
|
|
|
}
|
|
|
|
|
|
|
|
stateRoots := make([][]byte, params.BeaconConfig().SlotsPerHistoricalRoot)
|
|
|
|
for i := 0; i < len(stateRoots); i++ {
|
|
|
|
stateRoots[i] = zeroHash
|
|
|
|
}
|
|
|
|
|
|
|
|
slashings := make([]uint64, params.BeaconConfig().EpochsPerSlashingsVector)
|
|
|
|
|
|
|
|
genesisValidatorsRoot, err := stateutil.ValidatorRegistryRoot(preState.Validators())
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrapf(err, "could not hash tree root genesis validators %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
prevEpochParticipation, err := preState.PreviousEpochParticipation()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
currEpochParticipation, err := preState.CurrentEpochParticipation()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
scores, err := preState.InactivityScores()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
st := ðpb.BeaconStateDeneb{
|
|
|
|
// Misc fields.
|
|
|
|
Slot: 0,
|
|
|
|
GenesisTime: genesisTime,
|
|
|
|
GenesisValidatorsRoot: genesisValidatorsRoot[:],
|
|
|
|
|
|
|
|
Fork: ðpb.Fork{
|
|
|
|
PreviousVersion: params.BeaconConfig().GenesisForkVersion,
|
|
|
|
CurrentVersion: params.BeaconConfig().GenesisForkVersion,
|
|
|
|
Epoch: 0,
|
|
|
|
},
|
|
|
|
|
|
|
|
// Validator registry fields.
|
|
|
|
Validators: preState.Validators(),
|
|
|
|
Balances: preState.Balances(),
|
|
|
|
PreviousEpochParticipation: prevEpochParticipation,
|
|
|
|
CurrentEpochParticipation: currEpochParticipation,
|
|
|
|
InactivityScores: scores,
|
|
|
|
|
|
|
|
// Randomness and committees.
|
|
|
|
RandaoMixes: randaoMixes,
|
|
|
|
|
|
|
|
// Finality.
|
|
|
|
PreviousJustifiedCheckpoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 0,
|
|
|
|
Root: params.BeaconConfig().ZeroHash[:],
|
|
|
|
},
|
|
|
|
CurrentJustifiedCheckpoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 0,
|
|
|
|
Root: params.BeaconConfig().ZeroHash[:],
|
|
|
|
},
|
|
|
|
JustificationBits: []byte{0},
|
|
|
|
FinalizedCheckpoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 0,
|
|
|
|
Root: params.BeaconConfig().ZeroHash[:],
|
|
|
|
},
|
|
|
|
|
|
|
|
HistoricalRoots: [][]byte{},
|
|
|
|
BlockRoots: blockRoots,
|
|
|
|
StateRoots: stateRoots,
|
|
|
|
Slashings: slashings,
|
|
|
|
|
|
|
|
// Eth1 data.
|
|
|
|
Eth1Data: eth1Data,
|
|
|
|
Eth1DataVotes: []*ethpb.Eth1Data{},
|
|
|
|
Eth1DepositIndex: preState.Eth1DepositIndex(),
|
|
|
|
}
|
|
|
|
|
|
|
|
var scBits [fieldparams.SyncAggregateSyncCommitteeBytesLength]byte
|
|
|
|
bodyRoot, err := (ðpb.BeaconBlockBodyDeneb{
|
|
|
|
RandaoReveal: make([]byte, 96),
|
|
|
|
Eth1Data: ðpb.Eth1Data{
|
|
|
|
DepositRoot: make([]byte, 32),
|
|
|
|
BlockHash: make([]byte, 32),
|
|
|
|
},
|
|
|
|
Graffiti: make([]byte, 32),
|
|
|
|
SyncAggregate: ðpb.SyncAggregate{
|
|
|
|
SyncCommitteeBits: scBits[:],
|
|
|
|
SyncCommitteeSignature: make([]byte, 96),
|
|
|
|
},
|
|
|
|
ExecutionPayload: &enginev1.ExecutionPayloadDeneb{
|
|
|
|
ParentHash: make([]byte, 32),
|
|
|
|
FeeRecipient: make([]byte, 20),
|
|
|
|
StateRoot: make([]byte, 32),
|
|
|
|
ReceiptsRoot: make([]byte, 32),
|
|
|
|
LogsBloom: make([]byte, 256),
|
|
|
|
PrevRandao: make([]byte, 32),
|
2024-01-09 21:49:35 +00:00
|
|
|
ExtraData: make([]byte, 0),
|
2023-05-09 21:38:13 +00:00
|
|
|
BaseFeePerGas: make([]byte, 32),
|
|
|
|
BlockHash: make([]byte, 32),
|
2024-01-09 21:49:35 +00:00
|
|
|
Transactions: make([][]byte, 0),
|
|
|
|
Withdrawals: make([]*enginev1.Withdrawal, 0),
|
2023-05-09 21:38:13 +00:00
|
|
|
},
|
|
|
|
}).HashTreeRoot()
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not hash tree root empty block body")
|
|
|
|
}
|
|
|
|
|
|
|
|
st.LatestBlockHeader = ðpb.BeaconBlockHeader{
|
|
|
|
ParentRoot: zeroHash,
|
|
|
|
StateRoot: zeroHash,
|
|
|
|
BodyRoot: bodyRoot[:],
|
|
|
|
}
|
|
|
|
|
|
|
|
var pubKeys [][]byte
|
|
|
|
vals := preState.Validators()
|
|
|
|
for i := uint64(0); i < params.BeaconConfig().SyncCommitteeSize; i++ {
|
|
|
|
j := i % uint64(len(vals))
|
|
|
|
pubKeys = append(pubKeys, vals[j].PublicKey)
|
|
|
|
}
|
|
|
|
aggregated, err := bls.AggregatePublicKeys(pubKeys)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
st.CurrentSyncCommittee = ðpb.SyncCommittee{
|
|
|
|
Pubkeys: pubKeys,
|
|
|
|
AggregatePubkey: aggregated.Marshal(),
|
|
|
|
}
|
|
|
|
st.NextSyncCommittee = ðpb.SyncCommittee{
|
|
|
|
Pubkeys: pubKeys,
|
|
|
|
AggregatePubkey: aggregated.Marshal(),
|
|
|
|
}
|
|
|
|
|
|
|
|
st.LatestExecutionPayloadHeader = &enginev1.ExecutionPayloadHeaderDeneb{
|
|
|
|
ParentHash: make([]byte, 32),
|
|
|
|
FeeRecipient: make([]byte, 20),
|
|
|
|
StateRoot: make([]byte, 32),
|
|
|
|
ReceiptsRoot: make([]byte, 32),
|
|
|
|
LogsBloom: make([]byte, 256),
|
|
|
|
PrevRandao: make([]byte, 32),
|
2024-01-09 21:49:35 +00:00
|
|
|
ExtraData: make([]byte, 0),
|
2023-05-09 21:38:13 +00:00
|
|
|
BaseFeePerGas: make([]byte, 32),
|
|
|
|
BlockHash: make([]byte, 32),
|
|
|
|
TransactionsRoot: make([]byte, 32),
|
|
|
|
WithdrawalsRoot: make([]byte, 32),
|
|
|
|
}
|
|
|
|
|
|
|
|
return state_native.InitializeFromProtoDeneb(st)
|
|
|
|
}
|