2022-08-19 09:26:25 +00:00
package testnet
import (
"context"
"encoding/hex"
"encoding/json"
"fmt"
2023-03-15 12:31:57 +00:00
"math/big"
2022-08-19 09:26:25 +00:00
"os"
"strings"
2023-04-13 17:19:06 +00:00
"time"
2022-08-19 09:26:25 +00:00
2023-04-13 17:19:06 +00:00
"github.com/ethereum/go-ethereum/core"
2023-03-15 12:31:57 +00:00
"github.com/ethereum/go-ethereum/ethclient"
"github.com/ethereum/go-ethereum/rpc"
"github.com/ghodss/yaml"
"github.com/pkg/errors"
2023-03-17 18:52:56 +00:00
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
"github.com/prysmaticlabs/prysm/v4/cmd/flags"
"github.com/prysmaticlabs/prysm/v4/config/params"
"github.com/prysmaticlabs/prysm/v4/container/trie"
"github.com/prysmaticlabs/prysm/v4/io/file"
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
"github.com/prysmaticlabs/prysm/v4/runtime/interop"
"github.com/prysmaticlabs/prysm/v4/runtime/version"
2022-08-19 09:26:25 +00:00
"github.com/sirupsen/logrus"
"github.com/urfave/cli/v2"
)
var (
generateGenesisStateFlags = struct {
2023-04-13 17:19:06 +00:00
DepositJsonFile string
ChainConfigFile string
ConfigName string
NumValidators uint64
GenesisTime uint64
OutputSSZ string
OutputJSON string
OutputYaml string
ForkName string
OverrideEth1Data bool
ExecutionEndpoint string
GethGenesisJsonIn string
GethGenesisJsonOut string
2022-08-19 09:26:25 +00:00
} { }
log = logrus . WithField ( "prefix" , "genesis" )
outputSSZFlag = & cli . StringFlag {
Name : "output-ssz" ,
Destination : & generateGenesisStateFlags . OutputSSZ ,
Usage : "Output filename of the SSZ marshaling of the generated genesis state" ,
Value : "" ,
}
outputYamlFlag = & cli . StringFlag {
Name : "output-yaml" ,
Destination : & generateGenesisStateFlags . OutputYaml ,
Usage : "Output filename of the YAML marshaling of the generated genesis state" ,
Value : "" ,
}
outputJsonFlag = & cli . StringFlag {
Name : "output-json" ,
Destination : & generateGenesisStateFlags . OutputJSON ,
Usage : "Output filename of the JSON marshaling of the generated genesis state" ,
Value : "" ,
}
generateGenesisStateCmd = & cli . Command {
2022-12-13 12:13:33 +00:00
Name : "generate-genesis" ,
Usage : "Generate a beacon chain genesis state" ,
Action : func ( cliCtx * cli . Context ) error {
if err := cliActionGenerateGenesisState ( cliCtx ) ; err != nil {
log . WithError ( err ) . Fatal ( "Could not generate beacon chain genesis state" )
}
return nil
} ,
2022-08-19 09:26:25 +00:00
Flags : [ ] cli . Flag {
& cli . StringFlag {
Name : "chain-config-file" ,
Destination : & generateGenesisStateFlags . ChainConfigFile ,
Usage : "The path to a YAML file with chain config values" ,
} ,
& cli . StringFlag {
Name : "deposit-json-file" ,
Destination : & generateGenesisStateFlags . DepositJsonFile ,
Usage : "Path to deposit_data.json file generated by the staking-deposit-cli tool for optionally specifying validators in genesis state" ,
} ,
& cli . StringFlag {
Name : "config-name" ,
2023-03-01 05:26:32 +00:00
Usage : "Config kind to be used for generating the genesis state. Default: mainnet. Options include mainnet, interop, minimal, prater, sepolia. --chain-config-file will override this flag." ,
2022-08-19 09:26:25 +00:00
Destination : & generateGenesisStateFlags . ConfigName ,
Value : params . MainnetName ,
} ,
& cli . Uint64Flag {
Name : "num-validators" ,
Usage : "Number of validators to deterministically generate in the genesis state" ,
Destination : & generateGenesisStateFlags . NumValidators ,
Required : true ,
} ,
& cli . Uint64Flag {
Name : "genesis-time" ,
Destination : & generateGenesisStateFlags . GenesisTime ,
Usage : "Unix timestamp seconds used as the genesis time in the genesis state. If unset, defaults to now()" ,
} ,
2023-03-15 12:31:57 +00:00
& cli . BoolFlag {
Name : "override-eth1data" ,
Destination : & generateGenesisStateFlags . OverrideEth1Data ,
Usage : "Overrides Eth1Data with values from execution client. If unset, defaults to false" ,
Value : false ,
} ,
2023-04-13 17:19:06 +00:00
& cli . StringFlag {
Name : "geth-genesis-json-in" ,
Destination : & generateGenesisStateFlags . GethGenesisJsonIn ,
Usage : "Path to a \"genesis.json\" file, containing a json representation of Geth's core.Genesis" ,
} ,
& cli . StringFlag {
Name : "geth-genesis-json-out" ,
Destination : & generateGenesisStateFlags . GethGenesisJsonOut ,
Usage : "Path to write generated \"genesis.json\" file, containing a json representation of Geth's core.Genesis" ,
} ,
2023-03-15 12:31:57 +00:00
& cli . StringFlag {
Name : "execution-endpoint" ,
Destination : & generateGenesisStateFlags . ExecutionEndpoint ,
Usage : "Endpoint to preferred execution client. If unset, defaults to Geth" ,
Value : "http://localhost:8545" ,
} ,
2022-11-23 14:22:24 +00:00
flags . EnumValue {
Name : "fork" ,
Usage : fmt . Sprintf ( "Name of the BeaconState schema to use in output encoding [%s]" , strings . Join ( versionNames ( ) , "," ) ) ,
Enum : versionNames ( ) ,
Value : versionNames ( ) [ 0 ] ,
Destination : & generateGenesisStateFlags . ForkName ,
} . GenericFlag ( ) ,
2022-08-19 09:26:25 +00:00
outputSSZFlag ,
outputYamlFlag ,
outputJsonFlag ,
} ,
}
)
2022-11-23 14:22:24 +00:00
func versionNames ( ) [ ] string {
enum := version . All ( )
names := make ( [ ] string , len ( enum ) )
for i := range enum {
names [ i ] = version . String ( enum [ i ] )
}
return names
}
2022-08-19 09:26:25 +00:00
// Represents a json object of hex string and uint64 values for
// validators on Ethereum. This file can be generated using the official staking-deposit-cli.
type depositDataJSON struct {
PubKey string ` json:"pubkey" `
Amount uint64 ` json:"amount" `
WithdrawalCredentials string ` json:"withdrawal_credentials" `
DepositDataRoot string ` json:"deposit_data_root" `
Signature string ` json:"signature" `
}
func cliActionGenerateGenesisState ( cliCtx * cli . Context ) error {
outputJson := generateGenesisStateFlags . OutputJSON
outputYaml := generateGenesisStateFlags . OutputYaml
outputSSZ := generateGenesisStateFlags . OutputSSZ
noOutputFlag := outputSSZ == "" && outputJson == "" && outputYaml == ""
if noOutputFlag {
return fmt . Errorf (
"no %s, %s, %s flag(s) specified. At least one is required" ,
outputJsonFlag . Name ,
outputYamlFlag . Name ,
outputSSZFlag . Name ,
)
}
if err := setGlobalParams ( ) ; err != nil {
return fmt . Errorf ( "could not set config params: %v" , err )
}
2023-04-13 17:19:06 +00:00
st , err := generateGenesis ( cliCtx . Context )
2022-08-19 09:26:25 +00:00
if err != nil {
return fmt . Errorf ( "could not generate genesis state: %v" , err )
}
2022-11-23 14:22:24 +00:00
2022-08-19 09:26:25 +00:00
if outputJson != "" {
2022-11-23 14:22:24 +00:00
if err := writeToOutputFile ( outputJson , st , json . Marshal ) ; err != nil {
2022-08-19 09:26:25 +00:00
return err
}
}
if outputYaml != "" {
2022-11-23 14:22:24 +00:00
if err := writeToOutputFile ( outputYaml , st , yaml . Marshal ) ; err != nil {
2022-08-19 09:26:25 +00:00
return err
}
}
if outputSSZ != "" {
2022-11-23 14:22:24 +00:00
type MinimumSSZMarshal interface {
MarshalSSZ ( ) ( [ ] byte , error )
}
2022-08-19 09:26:25 +00:00
marshalFn := func ( o interface { } ) ( [ ] byte , error ) {
2022-11-23 14:22:24 +00:00
marshaler , ok := o . ( MinimumSSZMarshal )
2022-08-19 09:26:25 +00:00
if ! ok {
return nil , errors . New ( "not a marshaler" )
}
return marshaler . MarshalSSZ ( )
}
2022-11-23 14:22:24 +00:00
if err := writeToOutputFile ( outputSSZ , st , marshalFn ) ; err != nil {
2022-08-19 09:26:25 +00:00
return err
}
}
log . Info ( "Command completed" )
return nil
}
func setGlobalParams ( ) error {
chainConfigFile := generateGenesisStateFlags . ChainConfigFile
if chainConfigFile != "" {
log . Infof ( "Specified a chain config file: %s" , chainConfigFile )
return params . LoadChainConfigFile ( chainConfigFile , nil )
}
cfg , err := params . ByName ( generateGenesisStateFlags . ConfigName )
if err != nil {
return fmt . Errorf ( "unable to find config using name %s: %v" , generateGenesisStateFlags . ConfigName , err )
}
return params . SetActive ( cfg . Copy ( ) )
}
2023-04-13 17:19:06 +00:00
func generateGenesis ( ctx context . Context ) ( state . BeaconState , error ) {
f := & generateGenesisStateFlags
if f . GenesisTime == 0 {
f . GenesisTime = uint64 ( time . Now ( ) . Unix ( ) )
log . Info ( "No genesis time specified, defaulting to now()" )
}
v , err := version . FromString ( f . ForkName )
if err != nil {
return nil , err
}
opts := make ( [ ] interop . PremineGenesisOpt , 0 )
nv := f . NumValidators
if f . DepositJsonFile != "" {
expanded , err := file . ExpandPath ( f . DepositJsonFile )
2022-08-19 09:26:25 +00:00
if err != nil {
return nil , err
}
2023-04-13 17:19:06 +00:00
log . Printf ( "reading deposits from JSON at %s" , expanded )
b , err := os . ReadFile ( expanded ) // #nosec G304
2022-08-19 09:26:25 +00:00
if err != nil {
return nil , err
}
2023-04-13 17:19:06 +00:00
roots , dds , err := depositEntriesFromJSON ( b )
2023-04-03 15:13:51 +00:00
if err != nil {
return nil , err
}
2023-04-13 17:19:06 +00:00
opts = append ( opts , interop . WithDepositData ( dds , roots ) )
} else if nv == 0 {
return nil , fmt . Errorf (
"expected --num-validators > 0 or --deposit-json-file to have been provided" ,
)
}
gen := & core . Genesis { }
if f . GethGenesisJsonIn != "" {
gbytes , err := os . ReadFile ( f . GethGenesisJsonIn ) // #nosec G304
if err != nil {
return nil , errors . Wrapf ( err , "failed to read %s" , f . GethGenesisJsonIn )
2023-04-03 15:13:51 +00:00
}
2023-04-13 17:19:06 +00:00
if err := json . Unmarshal ( gbytes , gen ) ; err != nil {
return nil , err
}
2023-04-14 20:32:40 +00:00
// set timestamps for genesis and shanghai fork
gen . Timestamp = f . GenesisTime
gen . Config . ShanghaiTime = interop . GethShanghaiTime ( f . GenesisTime , params . BeaconConfig ( ) )
2023-05-03 04:34:01 +00:00
log .
WithField ( "shanghai" , gen . Config . ShanghaiTime ) .
Info ( "setting fork geth times" )
2023-04-14 20:32:40 +00:00
if v > version . Altair {
// set ttd to zero so EL goes post-merge immediately
gen . Config . TerminalTotalDifficulty = big . NewInt ( 0 )
}
2023-04-13 17:19:06 +00:00
} else {
gen = interop . GethTestnetGenesis ( f . GenesisTime , params . BeaconConfig ( ) )
}
2023-04-14 20:32:40 +00:00
2023-04-13 17:19:06 +00:00
if f . GethGenesisJsonOut != "" {
gbytes , err := json . MarshalIndent ( gen , "" , "\t" )
2023-04-03 15:13:51 +00:00
if err != nil {
return nil , err
}
2023-04-13 17:19:06 +00:00
if err := os . WriteFile ( f . GethGenesisJsonOut , gbytes , os . ModePerm ) ; err != nil {
return nil , errors . Wrapf ( err , "failed to write %s" , f . GethGenesisJsonOut )
}
}
gb := gen . ToBlock ( )
// TODO: expose the PregenesisCreds option with a cli flag - for now defaulting to no withdrawal credentials at genesis
genesisState , err := interop . NewPreminedGenesis ( ctx , f . GenesisTime , nv , 0 , v , gb , opts ... )
if err != nil {
return nil , err
2022-08-19 09:26:25 +00:00
}
2023-04-13 17:19:06 +00:00
if f . OverrideEth1Data {
2023-03-15 12:31:57 +00:00
log . Print ( "Overriding Eth1Data with data from execution client" )
conn , err := rpc . Dial ( generateGenesisStateFlags . ExecutionEndpoint )
if err != nil {
return nil , errors . Wrapf (
err ,
"could not dial %s please make sure you are running your execution client" ,
generateGenesisStateFlags . ExecutionEndpoint )
}
client := ethclient . NewClient ( conn )
header , err := client . HeaderByNumber ( ctx , big . NewInt ( 0 ) )
if err != nil {
return nil , errors . Wrap ( err , "could not get header by number" )
}
t , err := trie . NewTrie ( params . BeaconConfig ( ) . DepositContractTreeDepth )
if err != nil {
return nil , errors . Wrap ( err , "could not create deposit tree" )
}
depositRoot , err := t . HashTreeRoot ( )
if err != nil {
return nil , errors . Wrap ( err , "could not get hash tree root" )
}
2023-04-13 17:19:06 +00:00
e1d := & ethpb . Eth1Data {
2023-03-15 12:31:57 +00:00
DepositRoot : depositRoot [ : ] ,
DepositCount : 0 ,
BlockHash : header . Hash ( ) . Bytes ( ) ,
}
2023-04-13 17:19:06 +00:00
if err := genesisState . SetEth1Data ( e1d ) ; err != nil {
return nil , err
}
if err := genesisState . SetEth1DepositIndex ( 0 ) ; err != nil {
return nil , err
}
2023-03-15 12:31:57 +00:00
}
2023-04-13 17:19:06 +00:00
2022-08-19 09:26:25 +00:00
return genesisState , err
}
2023-04-13 17:19:06 +00:00
func depositEntriesFromJSON ( enc [ ] byte ) ( [ ] [ ] byte , [ ] * ethpb . Deposit_Data , error ) {
2022-08-19 09:26:25 +00:00
var depositJSON [ ] * depositDataJSON
if err := json . Unmarshal ( enc , & depositJSON ) ; err != nil {
2023-04-13 17:19:06 +00:00
return nil , nil , err
2022-08-19 09:26:25 +00:00
}
2023-04-13 17:19:06 +00:00
dds := make ( [ ] * ethpb . Deposit_Data , len ( depositJSON ) )
roots := make ( [ ] [ ] byte , len ( depositJSON ) )
2022-08-19 09:26:25 +00:00
for i , val := range depositJSON {
2023-04-13 17:19:06 +00:00
root , data , err := depositJSONToDepositData ( val )
2022-08-19 09:26:25 +00:00
if err != nil {
2023-04-13 17:19:06 +00:00
return nil , nil , err
2022-08-19 09:26:25 +00:00
}
2023-04-13 17:19:06 +00:00
dds [ i ] = data
roots [ i ] = root
2022-08-19 09:26:25 +00:00
}
2023-04-13 17:19:06 +00:00
return roots , dds , nil
2022-08-19 09:26:25 +00:00
}
2023-04-13 17:19:06 +00:00
func depositJSONToDepositData ( input * depositDataJSON ) ( [ ] byte , * ethpb . Deposit_Data , error ) {
root , err := hex . DecodeString ( strings . TrimPrefix ( input . DepositDataRoot , "0x" ) )
2022-08-19 09:26:25 +00:00
if err != nil {
2023-04-13 17:19:06 +00:00
return nil , nil , err
2022-08-19 09:26:25 +00:00
}
2023-04-13 17:19:06 +00:00
pk , err := hex . DecodeString ( strings . TrimPrefix ( input . PubKey , "0x" ) )
2022-08-19 09:26:25 +00:00
if err != nil {
2023-04-13 17:19:06 +00:00
return nil , nil , err
2022-08-19 09:26:25 +00:00
}
2023-04-13 17:19:06 +00:00
creds , err := hex . DecodeString ( strings . TrimPrefix ( input . WithdrawalCredentials , "0x" ) )
2022-08-19 09:26:25 +00:00
if err != nil {
2023-04-13 17:19:06 +00:00
return nil , nil , err
2022-08-19 09:26:25 +00:00
}
2023-04-13 17:19:06 +00:00
sig , err := hex . DecodeString ( strings . TrimPrefix ( input . Signature , "0x" ) )
2022-08-19 09:26:25 +00:00
if err != nil {
2023-04-13 17:19:06 +00:00
return nil , nil , err
2022-08-19 09:26:25 +00:00
}
2023-04-13 17:19:06 +00:00
return root , & ethpb . Deposit_Data {
PublicKey : pk ,
WithdrawalCredentials : creds ,
2022-08-19 09:26:25 +00:00
Amount : input . Amount ,
2023-04-13 17:19:06 +00:00
Signature : sig ,
} , nil
2022-08-19 09:26:25 +00:00
}
func writeToOutputFile (
fPath string ,
data interface { } ,
marshalFn func ( o interface { } ) ( [ ] byte , error ) ,
) error {
encoded , err := marshalFn ( data )
if err != nil {
return err
}
if err := file . WriteFile ( fPath , encoded ) ; err != nil {
return err
}
log . Printf ( "Done writing genesis state to %s" , fPath )
return nil
}