2020-10-15 22:31:52 +00:00
|
|
|
package accounts
|
2020-07-15 04:05:21 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2021-01-12 16:52:01 +00:00
|
|
|
"encoding/json"
|
2020-08-31 19:46:45 +00:00
|
|
|
"fmt"
|
2020-10-27 20:51:29 +00:00
|
|
|
"os"
|
|
|
|
"strings"
|
2020-07-15 04:05:21 +00:00
|
|
|
|
2020-08-31 19:46:45 +00:00
|
|
|
"github.com/manifoldco/promptui"
|
2020-07-15 04:05:21 +00:00
|
|
|
"github.com/pkg/errors"
|
2021-03-02 18:58:40 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/cmd/validator/flags"
|
2021-09-17 21:55:24 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/io/prompt"
|
2021-02-24 18:05:46 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/validator/accounts/iface"
|
2021-09-17 21:55:24 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/validator/accounts/userprompt"
|
2020-10-15 22:31:52 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/validator/accounts/wallet"
|
|
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager"
|
|
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager/derived"
|
2022-02-01 19:54:19 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager/local"
|
2020-10-15 22:31:52 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager/remote"
|
2022-06-27 13:34:38 +00:00
|
|
|
remoteweb3signer "github.com/prysmaticlabs/prysm/validator/keymanager/remote-web3signer"
|
2020-07-15 04:05:21 +00:00
|
|
|
"github.com/urfave/cli/v2"
|
|
|
|
)
|
|
|
|
|
2020-08-31 19:46:45 +00:00
|
|
|
// CreateWalletConfig defines the parameters needed to call the create wallet functions.
|
|
|
|
type CreateWalletConfig struct {
|
2022-01-31 16:44:17 +00:00
|
|
|
SkipMnemonicConfirm bool
|
|
|
|
NumAccounts int
|
|
|
|
RemoteKeymanagerOpts *remote.KeymanagerOpts
|
2022-06-27 13:34:38 +00:00
|
|
|
Web3SignerSetupConfig *remoteweb3signer.SetupConfig
|
2022-01-31 16:44:17 +00:00
|
|
|
WalletCfg *wallet.Config
|
|
|
|
Mnemonic25thWord string
|
2020-08-31 19:46:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// CreateAndSaveWalletCli from user input with a desired keymanager. If a
|
2020-07-15 04:05:21 +00:00
|
|
|
// wallet already exists in the path, it suggests the user alternatives
|
|
|
|
// such as how to edit their existing wallet configuration.
|
2020-09-17 01:34:42 +00:00
|
|
|
func CreateAndSaveWalletCli(cliCtx *cli.Context) (*wallet.Wallet, error) {
|
2020-08-31 19:46:45 +00:00
|
|
|
keymanagerKind, err := extractKeymanagerKindFromCli(cliCtx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-06-16 14:14:03 +00:00
|
|
|
createWalletConfig, err := ExtractWalletCreationConfigFromCli(cliCtx, keymanagerKind)
|
2020-07-24 00:43:01 +00:00
|
|
|
if err != nil {
|
2020-07-29 01:20:13 +00:00
|
|
|
return nil, err
|
2020-07-24 00:43:01 +00:00
|
|
|
}
|
2020-09-18 21:01:46 +00:00
|
|
|
|
|
|
|
dir := createWalletConfig.WalletCfg.WalletDir
|
2020-09-30 14:13:37 +00:00
|
|
|
dirExists, err := wallet.Exists(dir)
|
2020-09-18 21:01:46 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-09-22 19:02:13 +00:00
|
|
|
if dirExists {
|
|
|
|
return nil, errors.New("a wallet already exists at this location. Please input an" +
|
2020-09-18 21:01:46 +00:00
|
|
|
" alternative location for the new wallet or remove the current wallet")
|
|
|
|
}
|
2020-09-30 14:13:37 +00:00
|
|
|
|
2020-09-22 14:49:07 +00:00
|
|
|
w, err := CreateWalletWithKeymanager(cliCtx.Context, createWalletConfig)
|
|
|
|
if err != nil {
|
2020-10-22 19:45:31 +00:00
|
|
|
return nil, errors.Wrap(err, "could not create wallet")
|
2020-09-22 14:49:07 +00:00
|
|
|
}
|
|
|
|
return w, nil
|
2020-08-31 19:46:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// CreateWalletWithKeymanager specified by configuration options.
|
2020-09-17 01:34:42 +00:00
|
|
|
func CreateWalletWithKeymanager(ctx context.Context, cfg *CreateWalletConfig) (*wallet.Wallet, error) {
|
2020-09-23 08:59:49 +00:00
|
|
|
w := wallet.New(&wallet.Config{
|
2020-09-17 01:34:42 +00:00
|
|
|
WalletDir: cfg.WalletCfg.WalletDir,
|
|
|
|
KeymanagerKind: cfg.WalletCfg.KeymanagerKind,
|
|
|
|
WalletPassword: cfg.WalletCfg.WalletPassword,
|
|
|
|
})
|
2020-08-31 19:46:45 +00:00
|
|
|
var err error
|
2020-07-22 04:49:04 +00:00
|
|
|
switch w.KeymanagerKind() {
|
2022-02-01 19:54:19 +00:00
|
|
|
case keymanager.Local:
|
2022-06-16 14:14:03 +00:00
|
|
|
if err = CreateLocalKeymanagerWallet(ctx, w); err != nil {
|
2020-11-16 22:26:04 +00:00
|
|
|
return nil, errors.Wrap(err, "could not initialize wallet")
|
2020-07-15 04:05:21 +00:00
|
|
|
}
|
2022-01-31 16:44:17 +00:00
|
|
|
// TODO(#9883) - Remove this when we have a better way to handle this. should be safe to use for now.
|
2021-02-24 18:05:46 +00:00
|
|
|
km, err := w.InitializeKeymanager(ctx, iface.InitKeymanagerConfig{ListenForChanges: false})
|
2021-01-12 16:52:01 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, ErrCouldNotInitializeKeymanager)
|
|
|
|
}
|
2022-02-01 19:54:19 +00:00
|
|
|
localKm, ok := km.(*local.Keymanager)
|
2021-01-12 16:52:01 +00:00
|
|
|
if !ok {
|
|
|
|
return nil, errors.Wrap(err, ErrCouldNotInitializeKeymanager)
|
|
|
|
}
|
2022-02-01 19:54:19 +00:00
|
|
|
accountsKeystore, err := localKm.CreateAccountsKeystore(ctx, make([][]byte, 0), make([][]byte, 0))
|
2021-01-12 16:52:01 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
encodedAccounts, err := json.MarshalIndent(accountsKeystore, "", "\t")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-02-01 19:54:19 +00:00
|
|
|
if err = w.WriteFileAtPath(ctx, local.AccountsPath, local.AccountsKeystoreFileName, encodedAccounts); err != nil {
|
2021-01-12 16:52:01 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-09-17 01:34:42 +00:00
|
|
|
log.WithField("--wallet-dir", cfg.WalletCfg.WalletDir).Info(
|
2020-11-16 22:26:04 +00:00
|
|
|
"Successfully created wallet with ability to import keystores",
|
2020-07-15 04:05:21 +00:00
|
|
|
)
|
2020-10-15 22:31:52 +00:00
|
|
|
case keymanager.Derived:
|
2020-11-16 22:26:04 +00:00
|
|
|
if err = createDerivedKeymanagerWallet(
|
|
|
|
ctx,
|
|
|
|
w,
|
|
|
|
cfg.Mnemonic25thWord,
|
|
|
|
cfg.SkipMnemonicConfirm,
|
|
|
|
cfg.NumAccounts,
|
|
|
|
); err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not initialize wallet")
|
2020-07-21 02:05:23 +00:00
|
|
|
}
|
2020-09-17 01:34:42 +00:00
|
|
|
log.WithField("--wallet-dir", cfg.WalletCfg.WalletDir).Info(
|
2020-11-16 22:26:04 +00:00
|
|
|
"Successfully created HD wallet from mnemonic and regenerated accounts",
|
2020-07-21 02:05:23 +00:00
|
|
|
)
|
2020-10-15 22:31:52 +00:00
|
|
|
case keymanager.Remote:
|
2020-08-31 19:46:45 +00:00
|
|
|
if err = createRemoteKeymanagerWallet(ctx, w, cfg.RemoteKeymanagerOpts); err != nil {
|
2020-11-16 22:26:04 +00:00
|
|
|
return nil, errors.Wrap(err, "could not initialize wallet")
|
2020-07-15 04:05:21 +00:00
|
|
|
}
|
2020-09-17 01:34:42 +00:00
|
|
|
log.WithField("--wallet-dir", cfg.WalletCfg.WalletDir).Info(
|
2020-07-15 04:05:21 +00:00
|
|
|
"Successfully created wallet with remote keymanager configuration",
|
|
|
|
)
|
2022-01-31 16:44:17 +00:00
|
|
|
case keymanager.Web3Signer:
|
|
|
|
return nil, errors.New("web3signer keymanager does not require persistent wallets.")
|
2020-07-15 04:05:21 +00:00
|
|
|
default:
|
2021-01-22 20:21:34 +00:00
|
|
|
return nil, errors.Wrapf(err, errKeymanagerNotSupported, w.KeymanagerKind())
|
2020-07-15 04:05:21 +00:00
|
|
|
}
|
2020-07-29 01:20:13 +00:00
|
|
|
return w, nil
|
2020-07-15 04:05:21 +00:00
|
|
|
}
|
|
|
|
|
2020-10-15 22:31:52 +00:00
|
|
|
func extractKeymanagerKindFromCli(cliCtx *cli.Context) (keymanager.Kind, error) {
|
2020-08-31 19:46:45 +00:00
|
|
|
return inputKeymanagerKind(cliCtx)
|
|
|
|
}
|
|
|
|
|
2022-06-16 14:14:03 +00:00
|
|
|
// ExtractWalletCreationConfigFromCli prompts the user for wallet creation input.
|
|
|
|
func ExtractWalletCreationConfigFromCli(cliCtx *cli.Context, keymanagerKind keymanager.Kind) (*CreateWalletConfig, error) {
|
2021-09-17 21:55:24 +00:00
|
|
|
walletDir, err := userprompt.InputDirectory(cliCtx, userprompt.WalletDirPromptText, flags.WalletDirFlag)
|
2020-08-31 19:46:45 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-09-17 21:55:24 +00:00
|
|
|
walletPassword, err := prompt.InputPassword(
|
2020-08-31 19:46:45 +00:00
|
|
|
cliCtx,
|
|
|
|
flags.WalletPasswordFileFlag,
|
2020-09-17 01:34:42 +00:00
|
|
|
wallet.NewWalletPasswordPromptText,
|
|
|
|
wallet.ConfirmPasswordPromptText,
|
2020-08-31 19:46:45 +00:00
|
|
|
true, /* Should confirm password */
|
2021-09-17 21:55:24 +00:00
|
|
|
prompt.ValidatePasswordInput,
|
2020-08-31 19:46:45 +00:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
createWalletConfig := &CreateWalletConfig{
|
2020-09-17 01:34:42 +00:00
|
|
|
WalletCfg: &wallet.Config{
|
2020-08-31 19:46:45 +00:00
|
|
|
WalletDir: walletDir,
|
|
|
|
KeymanagerKind: keymanagerKind,
|
|
|
|
WalletPassword: walletPassword,
|
|
|
|
},
|
|
|
|
SkipMnemonicConfirm: cliCtx.Bool(flags.SkipDepositConfirmationFlag.Name),
|
|
|
|
}
|
2020-10-27 20:51:29 +00:00
|
|
|
skipMnemonic25thWord := cliCtx.IsSet(flags.SkipMnemonic25thWordCheckFlag.Name)
|
|
|
|
has25thWordFile := cliCtx.IsSet(flags.Mnemonic25thWordFileFlag.Name)
|
2020-11-16 22:26:04 +00:00
|
|
|
if keymanagerKind == keymanager.Derived {
|
|
|
|
numAccounts, err := inputNumAccounts(cliCtx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not get number of accounts to generate")
|
|
|
|
}
|
|
|
|
createWalletConfig.NumAccounts = int(numAccounts)
|
|
|
|
}
|
2020-10-27 20:51:29 +00:00
|
|
|
if keymanagerKind == keymanager.Derived && !skipMnemonic25thWord && !has25thWordFile {
|
2021-09-17 21:55:24 +00:00
|
|
|
resp, err := prompt.ValidatePrompt(
|
|
|
|
os.Stdin, newMnemonicPassphraseYesNoText, prompt.ValidateYesOrNo,
|
2020-10-27 20:51:29 +00:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not validate choice")
|
|
|
|
}
|
2021-01-04 16:46:22 +00:00
|
|
|
if strings.EqualFold(resp, "y") {
|
2021-09-17 21:55:24 +00:00
|
|
|
mnemonicPassphrase, err := prompt.InputPassword(
|
2020-10-27 20:51:29 +00:00
|
|
|
cliCtx,
|
|
|
|
flags.Mnemonic25thWordFileFlag,
|
|
|
|
newMnemonicPassphrasePromptText,
|
|
|
|
"Confirm mnemonic passphrase",
|
|
|
|
true, /* Should confirm password */
|
|
|
|
func(input string) error {
|
|
|
|
if strings.TrimSpace(input) == "" {
|
|
|
|
return errors.New("input cannot be empty")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
createWalletConfig.Mnemonic25thWord = mnemonicPassphrase
|
|
|
|
}
|
|
|
|
}
|
2020-10-15 22:31:52 +00:00
|
|
|
if keymanagerKind == keymanager.Remote {
|
2021-09-17 21:55:24 +00:00
|
|
|
opts, err := userprompt.InputRemoteKeymanagerConfig(cliCtx)
|
2020-08-31 19:46:45 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not input remote keymanager config")
|
|
|
|
}
|
|
|
|
createWalletConfig.RemoteKeymanagerOpts = opts
|
|
|
|
}
|
2022-01-31 16:44:17 +00:00
|
|
|
if keymanagerKind == keymanager.Web3Signer {
|
|
|
|
return nil, errors.New("web3signer keymanager does not require persistent wallets.")
|
|
|
|
}
|
2020-08-31 19:46:45 +00:00
|
|
|
return createWalletConfig, nil
|
|
|
|
}
|
|
|
|
|
2022-06-16 14:14:03 +00:00
|
|
|
func CreateLocalKeymanagerWallet(_ context.Context, wallet *wallet.Wallet) error {
|
2020-08-04 22:53:34 +00:00
|
|
|
if wallet == nil {
|
|
|
|
return errors.New("nil wallet")
|
|
|
|
}
|
2020-07-24 00:43:01 +00:00
|
|
|
if err := wallet.SaveWallet(); err != nil {
|
|
|
|
return errors.Wrap(err, "could not save wallet to disk")
|
|
|
|
}
|
2020-07-15 04:05:21 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-10-27 20:51:29 +00:00
|
|
|
func createDerivedKeymanagerWallet(
|
|
|
|
ctx context.Context,
|
|
|
|
wallet *wallet.Wallet,
|
|
|
|
mnemonicPassphrase string,
|
2020-11-16 22:26:04 +00:00
|
|
|
skipMnemonicConfirm bool,
|
|
|
|
numAccounts int,
|
2020-10-27 20:51:29 +00:00
|
|
|
) error {
|
2020-11-16 22:26:04 +00:00
|
|
|
if wallet == nil {
|
|
|
|
return errors.New("nil wallet")
|
2020-07-21 02:05:23 +00:00
|
|
|
}
|
2020-07-24 00:43:01 +00:00
|
|
|
if err := wallet.SaveWallet(); err != nil {
|
|
|
|
return errors.Wrap(err, "could not save wallet to disk")
|
|
|
|
}
|
2020-11-16 22:26:04 +00:00
|
|
|
km, err := derived.NewKeymanager(ctx, &derived.SetupConfig{
|
2021-02-24 18:05:46 +00:00
|
|
|
Wallet: wallet,
|
|
|
|
ListenForChanges: true,
|
2020-10-27 20:51:29 +00:00
|
|
|
})
|
2020-08-25 19:30:26 +00:00
|
|
|
if err != nil {
|
2020-11-16 22:26:04 +00:00
|
|
|
return errors.Wrap(err, "could not initialize HD keymanager")
|
|
|
|
}
|
|
|
|
mnemonic, err := derived.GenerateAndConfirmMnemonic(skipMnemonicConfirm)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not confirm mnemonic")
|
|
|
|
}
|
|
|
|
if err := km.RecoverAccountsFromMnemonic(ctx, mnemonic, mnemonicPassphrase, numAccounts); err != nil {
|
|
|
|
return errors.Wrap(err, "could not recover accounts from mnemonic")
|
2020-08-25 19:30:26 +00:00
|
|
|
}
|
2020-07-21 02:05:23 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-09-17 01:34:42 +00:00
|
|
|
func createRemoteKeymanagerWallet(ctx context.Context, wallet *wallet.Wallet, opts *remote.KeymanagerOpts) error {
|
2020-08-31 19:46:45 +00:00
|
|
|
keymanagerConfig, err := remote.MarshalOptionsFile(ctx, opts)
|
2020-07-15 04:05:21 +00:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not marshal config file")
|
|
|
|
}
|
2020-07-24 00:43:01 +00:00
|
|
|
if err := wallet.SaveWallet(); err != nil {
|
|
|
|
return errors.Wrap(err, "could not save wallet to disk")
|
|
|
|
}
|
2020-07-15 04:05:21 +00:00
|
|
|
if err := wallet.WriteKeymanagerConfigToDisk(ctx, keymanagerConfig); err != nil {
|
|
|
|
return errors.Wrap(err, "could not write keymanager config to disk")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2020-08-31 19:46:45 +00:00
|
|
|
|
2020-10-15 22:31:52 +00:00
|
|
|
func inputKeymanagerKind(cliCtx *cli.Context) (keymanager.Kind, error) {
|
2020-08-31 19:46:45 +00:00
|
|
|
if cliCtx.IsSet(flags.KeymanagerKindFlag.Name) {
|
2020-10-15 22:31:52 +00:00
|
|
|
return keymanager.ParseKind(cliCtx.String(flags.KeymanagerKindFlag.Name))
|
2020-08-31 19:46:45 +00:00
|
|
|
}
|
|
|
|
promptSelect := promptui.Select{
|
|
|
|
Label: "Select a type of wallet",
|
|
|
|
Items: []string{
|
2022-02-01 19:54:19 +00:00
|
|
|
wallet.KeymanagerKindSelections[keymanager.Local],
|
2020-10-15 22:31:52 +00:00
|
|
|
wallet.KeymanagerKindSelections[keymanager.Derived],
|
|
|
|
wallet.KeymanagerKindSelections[keymanager.Remote],
|
2022-01-31 16:44:17 +00:00
|
|
|
wallet.KeymanagerKindSelections[keymanager.Web3Signer],
|
2020-08-31 19:46:45 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
selection, _, err := promptSelect.Run()
|
|
|
|
if err != nil {
|
2022-02-01 19:54:19 +00:00
|
|
|
return keymanager.Local, fmt.Errorf("could not select wallet type: %w", userprompt.FormatPromptError(err))
|
2020-08-31 19:46:45 +00:00
|
|
|
}
|
2020-10-15 22:31:52 +00:00
|
|
|
return keymanager.Kind(selection), nil
|
2020-08-31 19:46:45 +00:00
|
|
|
}
|