2020-07-01 21:30:01 +00:00
|
|
|
package v2
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
2020-08-10 18:14:10 +00:00
|
|
|
"strings"
|
2020-07-01 21:30:01 +00:00
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2020-10-07 02:15:54 +00:00
|
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/validator/accounts/v2"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2020-09-17 01:34:42 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/validator/accounts/v2/wallet"
|
2020-07-01 21:30:01 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/validator/flags"
|
|
|
|
v2keymanager "github.com/prysmaticlabs/prysm/validator/keymanager/v2"
|
2020-07-22 04:49:04 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager/v2/derived"
|
|
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager/v2/direct"
|
2020-07-01 21:30:01 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
"github.com/urfave/cli/v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
var log = logrus.WithField("prefix", "accounts-v2")
|
|
|
|
|
2020-08-31 19:46:45 +00:00
|
|
|
// CreateAccountConfig to run the create account function.
|
|
|
|
type CreateAccountConfig struct {
|
2020-09-17 01:34:42 +00:00
|
|
|
Wallet *wallet.Wallet
|
2020-08-31 19:46:45 +00:00
|
|
|
NumAccounts int64
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateAccountCli creates a new validator account from user input by opening
|
|
|
|
// a wallet from the user's specified path. This uses the CLI to extract information
|
|
|
|
// to perform account creation.
|
|
|
|
func CreateAccountCli(cliCtx *cli.Context) error {
|
2020-09-17 01:34:42 +00:00
|
|
|
w, err := wallet.OpenWalletOrElseCli(cliCtx, CreateAndSaveWalletCli)
|
2020-07-28 16:06:18 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-08-31 19:46:45 +00:00
|
|
|
numAccounts := cliCtx.Int64(flags.NumAccountsFlag.Name)
|
|
|
|
log.Info("Creating a new account...")
|
|
|
|
return CreateAccount(cliCtx.Context, &CreateAccountConfig{
|
2020-09-17 01:34:42 +00:00
|
|
|
Wallet: w,
|
2020-08-31 19:46:45 +00:00
|
|
|
NumAccounts: numAccounts,
|
|
|
|
})
|
|
|
|
}
|
2020-08-20 19:14:03 +00:00
|
|
|
|
2020-08-31 19:46:45 +00:00
|
|
|
// CreateAccount creates a new validator account from user input by opening
|
|
|
|
// a wallet from the user's specified path.
|
|
|
|
func CreateAccount(ctx context.Context, cfg *CreateAccountConfig) error {
|
|
|
|
keymanager, err := cfg.Wallet.InitializeKeymanager(ctx, false /* skip mnemonic confirm */)
|
2020-08-10 18:14:10 +00:00
|
|
|
if err != nil && strings.Contains(err.Error(), "invalid checksum") {
|
|
|
|
return errors.New("wrong wallet password entered")
|
|
|
|
}
|
2020-07-01 21:30:01 +00:00
|
|
|
if err != nil {
|
2020-07-22 02:04:08 +00:00
|
|
|
return errors.Wrap(err, "could not initialize keymanager")
|
2020-07-01 21:30:01 +00:00
|
|
|
}
|
2020-08-31 19:46:45 +00:00
|
|
|
switch cfg.Wallet.KeymanagerKind() {
|
2020-07-22 04:49:04 +00:00
|
|
|
case v2keymanager.Remote:
|
|
|
|
return errors.New("cannot create a new account for a remote keymanager")
|
|
|
|
case v2keymanager.Direct:
|
|
|
|
km, ok := keymanager.(*direct.Keymanager)
|
|
|
|
if !ok {
|
|
|
|
return errors.New("not a direct keymanager")
|
|
|
|
}
|
|
|
|
// Create a new validator account using the specified keymanager.
|
2020-10-02 21:35:28 +00:00
|
|
|
if _, _, err := km.CreateAccount(ctx); err != nil {
|
2020-07-22 04:49:04 +00:00
|
|
|
return errors.Wrap(err, "could not create account in wallet")
|
|
|
|
}
|
|
|
|
case v2keymanager.Derived:
|
|
|
|
km, ok := keymanager.(*derived.Keymanager)
|
|
|
|
if !ok {
|
|
|
|
return errors.New("not a derived keymanager")
|
|
|
|
}
|
2020-10-12 08:11:05 +00:00
|
|
|
startNum := km.NextAccountNumber()
|
2020-08-31 19:46:45 +00:00
|
|
|
if cfg.NumAccounts == 1 {
|
2020-10-02 21:35:28 +00:00
|
|
|
if _, _, err := km.CreateAccount(ctx); err != nil {
|
2020-07-27 14:03:30 +00:00
|
|
|
return errors.Wrap(err, "could not create account in wallet")
|
|
|
|
}
|
|
|
|
} else {
|
2020-08-31 19:46:45 +00:00
|
|
|
for i := 0; i < int(cfg.NumAccounts); i++ {
|
2020-10-02 21:35:28 +00:00
|
|
|
if _, _, err := km.CreateAccount(ctx); err != nil {
|
2020-07-27 14:03:30 +00:00
|
|
|
return errors.Wrap(err, "could not create account in wallet")
|
|
|
|
}
|
|
|
|
}
|
2020-08-31 19:46:45 +00:00
|
|
|
log.Infof(
|
|
|
|
"Successfully created %d accounts. Please use accounts-v2 list to view details for accounts %d through %d",
|
|
|
|
cfg.NumAccounts,
|
|
|
|
startNum,
|
|
|
|
startNum+uint64(cfg.NumAccounts)-1,
|
|
|
|
)
|
2020-07-22 04:49:04 +00:00
|
|
|
}
|
|
|
|
default:
|
2020-08-31 19:46:45 +00:00
|
|
|
return fmt.Errorf("keymanager kind %s not supported", cfg.Wallet.KeymanagerKind())
|
2020-07-01 21:30:01 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2020-10-07 02:15:54 +00:00
|
|
|
|
|
|
|
// DepositDataJSON creates a raw map to match the deposit_data.json file format
|
|
|
|
// from the official eth2.0-deposit-cli https://github.com/ethereum/eth2.0-deposit-cli.
|
|
|
|
// The reason we utilize this map is to ensure we match the format of
|
|
|
|
// the eth2 deposit cli, which utilizes snake case and hex strings to represent binary data.
|
|
|
|
// Our gRPC gateway instead uses camel case and base64, which is why we use this workaround.
|
|
|
|
func DepositDataJSON(depositData *ethpb.Deposit_Data) (map[string]string, error) {
|
|
|
|
depositMessage := &pb.DepositMessage{
|
|
|
|
Pubkey: depositData.PublicKey,
|
|
|
|
WithdrawalCredentials: depositData.WithdrawalCredentials,
|
|
|
|
Amount: depositData.Amount,
|
|
|
|
}
|
|
|
|
depositMessageRoot, err := depositMessage.HashTreeRoot()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
depositDataRoot, err := depositData.HashTreeRoot()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
data := make(map[string]string)
|
|
|
|
data["pubkey"] = fmt.Sprintf("%x", depositData.PublicKey)
|
|
|
|
data["withdrawal_credentials"] = fmt.Sprintf("%x", depositData.WithdrawalCredentials)
|
|
|
|
data["amount"] = fmt.Sprintf("%d", depositData.Amount)
|
|
|
|
data["signature"] = fmt.Sprintf("%x", depositData.Signature)
|
|
|
|
data["deposit_message_root"] = fmt.Sprintf("%x", depositMessageRoot)
|
|
|
|
data["deposit_data_root"] = fmt.Sprintf("%x", depositDataRoot)
|
|
|
|
data["fork_version"] = fmt.Sprintf("%x", params.BeaconConfig().GenesisForkVersion)
|
|
|
|
return data, nil
|
|
|
|
}
|