mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-10 19:51:20 +00:00
a019a0db4c
* combines func params * update leftovers Co-authored-by: prylabs-bulldozer[bot] <58059840+prylabs-bulldozer[bot]@users.noreply.github.com>
140 lines
5.0 KiB
Go
140 lines
5.0 KiB
Go
// Package depositutil contains useful functions for dealing
|
|
// with eth2 deposit inputs.
|
|
package depositutil
|
|
|
|
import (
|
|
"github.com/ethereum/go-ethereum/core/types"
|
|
"github.com/pkg/errors"
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
|
"github.com/prysmaticlabs/go-ssz"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/state"
|
|
contract "github.com/prysmaticlabs/prysm/contracts/deposit-contract"
|
|
p2ppb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
"github.com/prysmaticlabs/prysm/shared/bls"
|
|
"github.com/prysmaticlabs/prysm/shared/featureconfig"
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
)
|
|
|
|
// DepositInput for a given key. This input data can be used to when making a
|
|
// validator deposit. The input data includes a proof of possession field
|
|
// signed by the deposit key.
|
|
//
|
|
// Spec details about general deposit workflow:
|
|
// To submit a deposit:
|
|
//
|
|
// - Pack the validator's initialization parameters into deposit_data, a Deposit_Data SSZ object.
|
|
// - Let amount be the amount in Gwei to be deposited by the validator where MIN_DEPOSIT_AMOUNT <= amount <= MAX_EFFECTIVE_BALANCE.
|
|
// - Set deposit_data.amount = amount.
|
|
// - Let signature be the result of bls_sign of the signing_root(deposit_data) with domain=compute_domain(DOMAIN_DEPOSIT). (Deposits are valid regardless of fork version, compute_domain will default to zeroes there).
|
|
// - Send a transaction on the Ethereum 1.0 chain to DEPOSIT_CONTRACT_ADDRESS executing def deposit(pubkey: bytes[48], withdrawal_credentials: bytes[32], signature: bytes[96]) along with a deposit of amount Gwei.
|
|
//
|
|
// See: https://github.com/ethereum/eth2.0-specs/blob/master/specs/validator/0_beacon-chain-validator.md#submit-deposit
|
|
func DepositInput(depositKey, withdrawalKey bls.SecretKey, amountInGwei uint64) (*ethpb.Deposit_Data, [32]byte, error) {
|
|
di := ðpb.Deposit_Data{
|
|
PublicKey: depositKey.PublicKey().Marshal(),
|
|
WithdrawalCredentials: WithdrawalCredentialsHash(withdrawalKey),
|
|
Amount: amountInGwei,
|
|
}
|
|
|
|
sr, err := ssz.SigningRoot(di)
|
|
if err != nil {
|
|
return nil, [32]byte{}, err
|
|
}
|
|
|
|
domain, err := helpers.ComputeDomain(
|
|
params.BeaconConfig().DomainDeposit,
|
|
nil, /*forkVersion*/
|
|
nil, /*genesisValidatorsRoot*/
|
|
)
|
|
if err != nil {
|
|
return nil, [32]byte{}, err
|
|
}
|
|
root, err := (&p2ppb.SigningData{ObjectRoot: sr[:], Domain: domain}).HashTreeRoot()
|
|
if err != nil {
|
|
return nil, [32]byte{}, err
|
|
}
|
|
di.Signature = depositKey.Sign(root[:]).Marshal()
|
|
|
|
dr, err := di.HashTreeRoot()
|
|
if err != nil {
|
|
return nil, [32]byte{}, err
|
|
}
|
|
|
|
return di, dr, nil
|
|
}
|
|
|
|
// WithdrawalCredentialsHash forms a 32 byte hash of the withdrawal public
|
|
// address.
|
|
//
|
|
// The specification is as follows:
|
|
// withdrawal_credentials[:1] == BLS_WITHDRAWAL_PREFIX_BYTE
|
|
// withdrawal_credentials[1:] == hash(withdrawal_pubkey)[1:]
|
|
// where withdrawal_credentials is of type bytes32.
|
|
func WithdrawalCredentialsHash(withdrawalKey bls.SecretKey) []byte {
|
|
h := hashutil.Hash(withdrawalKey.PublicKey().Marshal())
|
|
return append([]byte{params.BeaconConfig().BLSWithdrawalPrefixByte}, h[1:]...)[:32]
|
|
}
|
|
|
|
// VerifyDepositSignature verifies the correctness of Eth1 deposit BLS signature
|
|
func VerifyDepositSignature(dd *ethpb.Deposit_Data, domain []byte) error {
|
|
if featureconfig.Get().SkipBLSVerify {
|
|
return nil
|
|
}
|
|
ddCopy := state.CopyDepositData(dd)
|
|
publicKey, err := bls.PublicKeyFromBytes(dd.PublicKey)
|
|
if err != nil {
|
|
return errors.Wrap(err, "could not convert bytes to public key")
|
|
}
|
|
sig, err := bls.SignatureFromBytes(dd.Signature)
|
|
if err != nil {
|
|
return errors.Wrap(err, "could not convert bytes to signature")
|
|
}
|
|
ddCopy.Signature = nil
|
|
root, err := ssz.SigningRoot(ddCopy)
|
|
if err != nil {
|
|
return errors.Wrap(err, "could not get signing root")
|
|
}
|
|
signingData := &p2ppb.SigningData{
|
|
ObjectRoot: root[:],
|
|
Domain: domain,
|
|
}
|
|
ctrRoot, err := signingData.HashTreeRoot()
|
|
if err != nil {
|
|
return errors.Wrap(err, "could not get container root")
|
|
}
|
|
if !sig.Verify(publicKey, ctrRoot[:]) {
|
|
return helpers.ErrSigFailedToVerify
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// GenerateDepositTransaction uses the provided validating key and withdrawal key to
|
|
// create a transaction object for the deposit contract.
|
|
func GenerateDepositTransaction(validatingKey, withdrawalKey bls.SecretKey) (*types.Transaction, *ethpb.Deposit_Data, error) {
|
|
depositData, depositRoot, err := DepositInput(
|
|
validatingKey, withdrawalKey, params.BeaconConfig().MaxEffectiveBalance,
|
|
)
|
|
if err != nil {
|
|
return nil, nil, errors.Wrap(err, "could not generate deposit input")
|
|
}
|
|
testAcc, err := contract.Setup()
|
|
if err != nil {
|
|
return nil, nil, errors.Wrap(err, "could not load deposit contract")
|
|
}
|
|
testAcc.TxOpts.GasLimit = 1000000
|
|
|
|
tx, err := testAcc.Contract.Deposit(
|
|
testAcc.TxOpts,
|
|
depositData.PublicKey,
|
|
depositData.WithdrawalCredentials,
|
|
depositData.Signature,
|
|
depositRoot,
|
|
)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
return tx, depositData, nil
|
|
}
|