2020-04-29 21:32:39 +00:00
|
|
|
// Package interop contains deterministic utilities for generating
|
|
|
|
// genesis states and keys.
|
2019-09-11 18:38:35 +00:00
|
|
|
package interop
|
|
|
|
|
|
|
|
import (
|
2021-04-05 15:07:56 +00:00
|
|
|
"context"
|
2019-11-03 21:25:52 +00:00
|
|
|
"sync"
|
|
|
|
|
2019-09-11 18:38:35 +00:00
|
|
|
"github.com/pkg/errors"
|
2024-02-15 05:46:47 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v5/async"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/core/signing"
|
|
|
|
coreState "github.com/prysmaticlabs/prysm/v5/beacon-chain/core/transition"
|
|
|
|
statenative "github.com/prysmaticlabs/prysm/v5/beacon-chain/state/state-native"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/config/params"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/container/trie"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/crypto/bls"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/crypto/hash"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/encoding/bytesutil"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/time"
|
2019-09-11 18:38:35 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2021-06-26 19:00:33 +00:00
|
|
|
// This is the recommended mock eth1 block hash according to the Ethereum consensus interop guidelines.
|
2019-09-11 18:38:35 +00:00
|
|
|
// https://github.com/ethereum/eth2.0-pm/blob/a085c9870f3956d6228ed2a40cd37f0c6580ecd7/interop/mocked_start/README.md
|
|
|
|
mockEth1BlockHash = []byte{66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66, 66}
|
|
|
|
)
|
|
|
|
|
|
|
|
// GenerateGenesisState deterministically given a genesis time and number of validators.
|
2019-09-27 15:49:55 +00:00
|
|
|
// If a genesis time of 0 is supplied it is set to the current time.
|
2021-07-29 21:45:17 +00:00
|
|
|
func GenerateGenesisState(ctx context.Context, genesisTime, numValidators uint64) (*ethpb.BeaconState, []*ethpb.Deposit, error) {
|
2019-09-11 18:38:35 +00:00
|
|
|
privKeys, pubKeys, err := DeterministicallyGenerateKeys(0 /*startIndex*/, numValidators)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrapf(err, "could not deterministically generate keys for %d validators", numValidators)
|
|
|
|
}
|
|
|
|
depositDataItems, depositDataRoots, err := DepositDataFromKeys(privKeys, pubKeys)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrap(err, "could not generate deposit data from keys")
|
|
|
|
}
|
2021-04-05 15:07:56 +00:00
|
|
|
return GenerateGenesisStateFromDepositData(ctx, genesisTime, depositDataItems, depositDataRoots)
|
2020-10-29 22:37:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GenerateGenesisStateFromDepositData creates a genesis state given a list of
|
|
|
|
// deposit data items and their corresponding roots.
|
|
|
|
func GenerateGenesisStateFromDepositData(
|
2021-04-05 15:07:56 +00:00
|
|
|
ctx context.Context, genesisTime uint64, depositData []*ethpb.Deposit_Data, depositDataRoots [][]byte,
|
2021-07-29 21:45:17 +00:00
|
|
|
) (*ethpb.BeaconState, []*ethpb.Deposit, error) {
|
2022-06-27 13:34:38 +00:00
|
|
|
t, err := trie.GenerateTrieFromItems(depositDataRoots, params.BeaconConfig().DepositContractTreeDepth)
|
2019-09-11 18:38:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrap(err, "could not generate Merkle trie for deposit proofs")
|
|
|
|
}
|
2022-06-27 13:34:38 +00:00
|
|
|
deposits, err := GenerateDepositsFromData(depositData, t)
|
2019-09-11 18:38:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrap(err, "could not generate deposits from the deposit data provided")
|
|
|
|
}
|
2022-06-27 13:34:38 +00:00
|
|
|
root, err := t.HashTreeRoot()
|
2022-05-09 16:51:48 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrap(err, "could not hash tree root of deposit trie")
|
|
|
|
}
|
2019-09-27 15:49:55 +00:00
|
|
|
if genesisTime == 0 {
|
2021-09-15 00:09:04 +00:00
|
|
|
genesisTime = uint64(time.Now().Unix())
|
2019-09-27 15:49:55 +00:00
|
|
|
}
|
2021-04-05 15:07:56 +00:00
|
|
|
beaconState, err := coreState.GenesisBeaconState(ctx, deposits, genesisTime, ðpb.Eth1Data{
|
2019-09-11 18:38:35 +00:00
|
|
|
DepositRoot: root[:],
|
|
|
|
DepositCount: uint64(len(deposits)),
|
|
|
|
BlockHash: mockEth1BlockHash,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrap(err, "could not generate genesis state")
|
|
|
|
}
|
2021-03-17 03:26:17 +00:00
|
|
|
|
2022-10-19 14:37:45 +00:00
|
|
|
pbState, err := statenative.ProtobufBeaconStatePhase0(beaconState.ToProtoUnsafe())
|
2021-03-17 03:26:17 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
return pbState, deposits, nil
|
2019-09-11 18:38:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GenerateDepositsFromData a list of deposit items by creating proofs for each of them from a sparse Merkle trie.
|
2021-09-16 17:05:58 +00:00
|
|
|
func GenerateDepositsFromData(depositDataItems []*ethpb.Deposit_Data, trie *trie.SparseMerkleTrie) ([]*ethpb.Deposit, error) {
|
2019-11-22 04:08:50 +00:00
|
|
|
deposits := make([]*ethpb.Deposit, len(depositDataItems))
|
2021-09-18 17:26:11 +00:00
|
|
|
results, err := async.Scatter(len(depositDataItems), func(offset int, entries int, _ *sync.RWMutex) (interface{}, error) {
|
2019-11-22 04:08:50 +00:00
|
|
|
return generateDepositsFromData(depositDataItems[offset:offset+entries], offset, trie)
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to generate deposits from data")
|
|
|
|
}
|
|
|
|
for _, result := range results {
|
|
|
|
if depositExtent, ok := result.Extent.([]*ethpb.Deposit); ok {
|
|
|
|
copy(deposits[result.Offset:], depositExtent)
|
|
|
|
} else {
|
|
|
|
return nil, errors.New("extent not of expected type")
|
2019-11-03 21:25:52 +00:00
|
|
|
}
|
|
|
|
}
|
2019-11-22 04:08:50 +00:00
|
|
|
return deposits, nil
|
2019-11-03 21:25:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// generateDepositsFromData a list of deposit items by creating proofs for each of them from a sparse Merkle trie.
|
2021-09-16 17:05:58 +00:00
|
|
|
func generateDepositsFromData(depositDataItems []*ethpb.Deposit_Data, offset int, trie *trie.SparseMerkleTrie) ([]*ethpb.Deposit, error) {
|
2019-09-11 18:38:35 +00:00
|
|
|
deposits := make([]*ethpb.Deposit, len(depositDataItems))
|
|
|
|
for i, item := range depositDataItems {
|
2019-11-03 21:25:52 +00:00
|
|
|
proof, err := trie.MerkleProof(i + offset)
|
2019-09-11 18:38:35 +00:00
|
|
|
if err != nil {
|
2019-11-03 21:25:52 +00:00
|
|
|
return nil, errors.Wrapf(err, "could not generate proof for deposit %d", i+offset)
|
2019-09-11 18:38:35 +00:00
|
|
|
}
|
|
|
|
deposits[i] = ðpb.Deposit{
|
|
|
|
Proof: proof,
|
|
|
|
Data: item,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return deposits, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DepositDataFromKeys generates a list of deposit data items from a set of BLS validator keys.
|
2020-06-25 00:47:51 +00:00
|
|
|
func DepositDataFromKeys(privKeys []bls.SecretKey, pubKeys []bls.PublicKey) ([]*ethpb.Deposit_Data, [][]byte, error) {
|
2019-11-22 04:08:50 +00:00
|
|
|
type depositData struct {
|
|
|
|
items []*ethpb.Deposit_Data
|
|
|
|
roots [][]byte
|
|
|
|
}
|
|
|
|
depositDataItems := make([]*ethpb.Deposit_Data, len(privKeys))
|
|
|
|
depositDataRoots := make([][]byte, len(privKeys))
|
2021-09-18 17:26:11 +00:00
|
|
|
results, err := async.Scatter(len(privKeys), func(offset int, entries int, _ *sync.RWMutex) (interface{}, error) {
|
2023-02-10 06:19:15 +00:00
|
|
|
items, roots, err := depositDataFromKeys(privKeys[offset:offset+entries], pubKeys[offset:offset+entries], 0)
|
2019-11-22 04:08:50 +00:00
|
|
|
return &depositData{items: items, roots: roots}, err
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrap(err, "failed to generate deposit data from keys")
|
|
|
|
}
|
|
|
|
for _, result := range results {
|
|
|
|
if depositDataExtent, ok := result.Extent.(*depositData); ok {
|
|
|
|
copy(depositDataItems[result.Offset:], depositDataExtent.items)
|
|
|
|
copy(depositDataRoots[result.Offset:], depositDataExtent.roots)
|
|
|
|
} else {
|
|
|
|
return nil, nil, errors.New("extent not of expected type")
|
2019-11-03 21:25:52 +00:00
|
|
|
}
|
|
|
|
}
|
2019-11-22 04:08:50 +00:00
|
|
|
return depositDataItems, depositDataRoots, nil
|
2019-11-03 21:25:52 +00:00
|
|
|
}
|
|
|
|
|
2023-02-10 06:19:15 +00:00
|
|
|
// DepositDataFromKeysWithExecCreds generates a list of deposit data items from a set of BLS validator keys.
|
|
|
|
func DepositDataFromKeysWithExecCreds(privKeys []bls.SecretKey, pubKeys []bls.PublicKey, numOfCreds uint64) ([]*ethpb.Deposit_Data, [][]byte, error) {
|
|
|
|
return depositDataFromKeys(privKeys, pubKeys, numOfCreds)
|
|
|
|
}
|
|
|
|
|
|
|
|
func depositDataFromKeys(privKeys []bls.SecretKey, pubKeys []bls.PublicKey, numOfCreds uint64) ([]*ethpb.Deposit_Data, [][]byte, error) {
|
2019-09-11 18:38:35 +00:00
|
|
|
dataRoots := make([][]byte, len(privKeys))
|
|
|
|
depositDataItems := make([]*ethpb.Deposit_Data, len(privKeys))
|
|
|
|
for i := 0; i < len(privKeys); i++ {
|
2023-02-10 06:19:15 +00:00
|
|
|
withCred := uint64(i) < numOfCreds
|
|
|
|
data, err := createDepositData(privKeys[i], pubKeys[i], withCred)
|
2019-09-11 18:38:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrapf(err, "could not create deposit data for key: %#x", privKeys[i].Marshal())
|
|
|
|
}
|
2022-06-27 13:34:38 +00:00
|
|
|
h, err := data.HashTreeRoot()
|
2019-09-11 18:38:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrap(err, "could not hash tree root deposit data item")
|
|
|
|
}
|
2022-06-27 13:34:38 +00:00
|
|
|
dataRoots[i] = h[:]
|
2019-09-11 18:38:35 +00:00
|
|
|
depositDataItems[i] = data
|
|
|
|
}
|
|
|
|
return depositDataItems, dataRoots, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generates a deposit data item from BLS keys and signs the hash tree root of the data.
|
2023-02-10 06:19:15 +00:00
|
|
|
func createDepositData(privKey bls.SecretKey, pubKey bls.PublicKey, withExecCreds bool) (*ethpb.Deposit_Data, error) {
|
2021-07-29 21:45:17 +00:00
|
|
|
depositMessage := ðpb.DepositMessage{
|
2019-09-11 18:38:35 +00:00
|
|
|
PublicKey: pubKey.Marshal(),
|
|
|
|
WithdrawalCredentials: withdrawalCredentialsHash(pubKey.Marshal()),
|
|
|
|
Amount: params.BeaconConfig().MaxEffectiveBalance,
|
|
|
|
}
|
2023-02-10 06:19:15 +00:00
|
|
|
if withExecCreds {
|
|
|
|
newCredentials := make([]byte, 12)
|
|
|
|
newCredentials[0] = params.BeaconConfig().ETH1AddressWithdrawalPrefixByte
|
|
|
|
execAddr := bytesutil.ToBytes20(pubKey.Marshal())
|
|
|
|
depositMessage.WithdrawalCredentials = append(newCredentials, execAddr[:]...)
|
|
|
|
}
|
2021-01-12 00:45:11 +00:00
|
|
|
sr, err := depositMessage.HashTreeRoot()
|
2019-09-11 18:38:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-09-27 16:19:20 +00:00
|
|
|
domain, err := signing.ComputeDomain(params.BeaconConfig().DomainDeposit, nil, nil)
|
2020-04-14 20:27:03 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-07-29 21:45:17 +00:00
|
|
|
root, err := (ðpb.SigningData{ObjectRoot: sr[:], Domain: domain}).HashTreeRoot()
|
2020-04-14 20:27:03 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-01-12 00:45:11 +00:00
|
|
|
di := ðpb.Deposit_Data{
|
|
|
|
PublicKey: depositMessage.PublicKey,
|
|
|
|
WithdrawalCredentials: depositMessage.WithdrawalCredentials,
|
|
|
|
Amount: depositMessage.Amount,
|
|
|
|
Signature: privKey.Sign(root[:]).Marshal(),
|
|
|
|
}
|
2019-09-11 18:38:35 +00:00
|
|
|
return di, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// withdrawalCredentialsHash forms a 32 byte hash of the withdrawal public
|
|
|
|
// address.
|
|
|
|
//
|
|
|
|
// The specification is as follows:
|
2022-11-18 19:12:19 +00:00
|
|
|
//
|
|
|
|
// withdrawal_credentials[:1] == BLS_WITHDRAWAL_PREFIX_BYTE
|
|
|
|
// withdrawal_credentials[1:] == hash(withdrawal_pubkey)[1:]
|
|
|
|
//
|
2019-09-11 18:38:35 +00:00
|
|
|
// where withdrawal_credentials is of type bytes32.
|
|
|
|
func withdrawalCredentialsHash(pubKey []byte) []byte {
|
2021-09-15 22:55:11 +00:00
|
|
|
h := hash.Hash(pubKey)
|
2019-09-18 14:52:34 +00:00
|
|
|
return append([]byte{blsWithdrawalPrefixByte}, h[1:]...)[:32]
|
2019-09-11 18:38:35 +00:00
|
|
|
}
|