2020-10-15 22:31:52 +00:00
|
|
|
package accounts
|
2020-07-22 02:41:39 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
2020-08-20 17:53:09 +00:00
|
|
|
"os"
|
2020-08-06 20:33:28 +00:00
|
|
|
"sort"
|
2020-07-27 14:03:30 +00:00
|
|
|
"strconv"
|
2020-07-22 02:41:39 +00:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
"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"
|
|
|
|
"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"
|
2020-08-06 20:33:28 +00:00
|
|
|
"github.com/tyler-smith/go-bip39"
|
|
|
|
"github.com/tyler-smith/go-bip39/wordlists"
|
2020-07-22 02:41:39 +00:00
|
|
|
"github.com/urfave/cli/v2"
|
|
|
|
)
|
|
|
|
|
2020-10-27 20:51:29 +00:00
|
|
|
const (
|
2021-08-15 15:24:13 +00:00
|
|
|
phraseWordCount = 24
|
2021-12-09 19:40:48 +00:00
|
|
|
// #nosec G101 -- Not sensitive data
|
2021-08-15 15:24:13 +00:00
|
|
|
newMnemonicPassphraseYesNoText = "(Advanced) Do you want to setup a '25th word' passphrase for your mnemonic? [y/n]"
|
2021-12-09 19:40:48 +00:00
|
|
|
// #nosec G101 -- Not sensitive data
|
2020-10-27 20:51:29 +00:00
|
|
|
newMnemonicPassphrasePromptText = "(Advanced) Setup a passphrase '25th word' for your mnemonic " +
|
|
|
|
"(WARNING: You cannot recover your keys from your mnemonic if you forget this passphrase!)"
|
2021-12-09 19:40:48 +00:00
|
|
|
// #nosec G101 -- Not sensitive data
|
2021-08-15 15:24:13 +00:00
|
|
|
mnemonicPassphraseYesNoText = "(Advanced) Do you have an optional '25th word' passphrase for your mnemonic? [y/n]"
|
2021-12-09 19:40:48 +00:00
|
|
|
// #nosec G101 -- Not sensitive data
|
2020-10-27 20:51:29 +00:00
|
|
|
mnemonicPassphrasePromptText = "(Advanced) Enter the '25th word' passphrase for your mnemonic"
|
|
|
|
)
|
2020-07-22 02:41:39 +00:00
|
|
|
|
2022-03-13 23:30:11 +00:00
|
|
|
var (
|
|
|
|
ErrIncorrectWordNumber = errors.New("incorrect number of words provided")
|
|
|
|
ErrEmptyMnemonic = errors.New("phrase cannot be empty")
|
|
|
|
)
|
|
|
|
|
2020-08-31 19:46:45 +00:00
|
|
|
// RecoverWalletConfig to run the recover wallet function.
|
|
|
|
type RecoverWalletConfig struct {
|
2020-10-27 20:51:29 +00:00
|
|
|
WalletDir string
|
|
|
|
WalletPassword string
|
|
|
|
Mnemonic string
|
2020-11-16 22:26:04 +00:00
|
|
|
NumAccounts int
|
2020-10-27 20:51:29 +00:00
|
|
|
Mnemonic25thWord string
|
2020-08-31 19:46:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// RecoverWalletCli uses a menmonic seed phrase to recover a wallet into the path provided. This
|
|
|
|
// uses the CLI to extract necessary values to run the function.
|
|
|
|
func RecoverWalletCli(cliCtx *cli.Context) error {
|
2020-07-22 02:41:39 +00:00
|
|
|
mnemonic, err := inputMnemonic(cliCtx)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not get mnemonic phrase")
|
|
|
|
}
|
2020-10-27 20:51:29 +00:00
|
|
|
config := &RecoverWalletConfig{
|
|
|
|
Mnemonic: mnemonic,
|
|
|
|
}
|
|
|
|
skipMnemonic25thWord := cliCtx.IsSet(flags.SkipMnemonic25thWordCheckFlag.Name)
|
|
|
|
has25thWordFile := cliCtx.IsSet(flags.Mnemonic25thWordFileFlag.Name)
|
|
|
|
if !skipMnemonic25thWord && !has25thWordFile {
|
2021-09-17 21:55:24 +00:00
|
|
|
resp, err := prompt.ValidatePrompt(
|
|
|
|
os.Stdin, mnemonicPassphraseYesNoText, prompt.ValidateYesOrNo,
|
2020-10-27 20:51:29 +00:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return 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,
|
|
|
|
mnemonicPassphrasePromptText,
|
|
|
|
"Confirm mnemonic passphrase",
|
|
|
|
false, /* Should confirm password */
|
|
|
|
func(input string) error {
|
|
|
|
if strings.TrimSpace(input) == "" {
|
|
|
|
return errors.New("input cannot be empty")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
config.Mnemonic25thWord = mnemonicPassphrase
|
|
|
|
}
|
|
|
|
}
|
2021-09-17 21:55:24 +00:00
|
|
|
walletDir, err := userprompt.InputDirectory(cliCtx, userprompt.WalletDirPromptText, flags.WalletDirFlag)
|
2020-07-22 02:41:39 +00:00
|
|
|
if err != nil {
|
2020-08-31 19:46:45 +00:00
|
|
|
return 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 err
|
|
|
|
}
|
|
|
|
numAccounts, err := inputNumAccounts(cliCtx)
|
2020-07-27 14:03:30 +00:00
|
|
|
if err != nil {
|
2020-08-31 19:46:45 +00:00
|
|
|
return errors.Wrap(err, "could not get number of accounts to recover")
|
2020-07-27 14:03:30 +00:00
|
|
|
}
|
2020-10-27 20:51:29 +00:00
|
|
|
config.WalletDir = walletDir
|
|
|
|
config.WalletPassword = walletPassword
|
2020-11-16 22:26:04 +00:00
|
|
|
config.NumAccounts = int(numAccounts)
|
|
|
|
if _, err = RecoverWallet(cliCtx.Context, config); err != nil {
|
2020-09-22 14:49:07 +00:00
|
|
|
return err
|
|
|
|
}
|
2020-08-31 19:46:45 +00:00
|
|
|
log.Infof(
|
2020-11-16 22:26:04 +00:00
|
|
|
"Successfully recovered HD wallet with accounts and saved configuration to disk",
|
2020-08-31 19:46:45 +00:00
|
|
|
)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// RecoverWallet uses a menmonic seed phrase to recover a wallet into the path provided.
|
2020-11-16 22:26:04 +00:00
|
|
|
func RecoverWallet(ctx context.Context, cfg *RecoverWalletConfig) (*wallet.Wallet, error) {
|
2020-09-30 14:13:37 +00:00
|
|
|
// Ensure that the wallet directory does not contain a wallet already
|
|
|
|
dirExists, err := wallet.Exists(cfg.WalletDir)
|
|
|
|
if err != nil {
|
2020-11-16 22:26:04 +00:00
|
|
|
return nil, err
|
2020-09-30 14:13:37 +00:00
|
|
|
}
|
|
|
|
if dirExists {
|
2020-11-16 22:26:04 +00:00
|
|
|
return nil, errors.New("a wallet already exists at this location. Please input an" +
|
2020-09-30 14:13:37 +00:00
|
|
|
" alternative location for the new wallet or remove the current wallet")
|
|
|
|
}
|
2020-09-23 08:59:49 +00:00
|
|
|
w := wallet.New(&wallet.Config{
|
2020-09-17 01:34:42 +00:00
|
|
|
WalletDir: cfg.WalletDir,
|
2020-10-15 22:31:52 +00:00
|
|
|
KeymanagerKind: keymanager.Derived,
|
2020-09-17 01:34:42 +00:00
|
|
|
WalletPassword: cfg.WalletPassword,
|
|
|
|
})
|
|
|
|
if err := w.SaveWallet(); err != nil {
|
2020-11-16 22:26:04 +00:00
|
|
|
return nil, errors.Wrap(err, "could not save wallet to disk")
|
2020-09-03 15:11:17 +00:00
|
|
|
}
|
2020-11-16 22:26:04 +00:00
|
|
|
km, err := derived.NewKeymanager(ctx, &derived.SetupConfig{
|
2021-02-24 18:05:46 +00:00
|
|
|
Wallet: w,
|
|
|
|
ListenForChanges: false,
|
2020-08-31 19:46:45 +00:00
|
|
|
})
|
2020-07-27 14:03:30 +00:00
|
|
|
if err != nil {
|
2020-11-16 22:26:04 +00:00
|
|
|
return nil, errors.Wrap(err, "could not make keymanager for given phrase")
|
2020-08-31 19:46:45 +00:00
|
|
|
}
|
2020-11-16 22:26:04 +00:00
|
|
|
if err := km.RecoverAccountsFromMnemonic(ctx, cfg.Mnemonic, cfg.Mnemonic25thWord, cfg.NumAccounts); err != nil {
|
|
|
|
return nil, err
|
2020-08-31 19:46:45 +00:00
|
|
|
}
|
2020-09-17 01:34:42 +00:00
|
|
|
log.WithField("wallet-path", w.AccountsDir()).Infof(
|
2020-12-17 13:37:06 +00:00
|
|
|
"Successfully recovered HD wallet with %d accounts. Please use `accounts list` to view details for your accounts",
|
2020-08-31 19:46:45 +00:00
|
|
|
cfg.NumAccounts,
|
2020-07-22 04:49:04 +00:00
|
|
|
)
|
2020-11-16 22:26:04 +00:00
|
|
|
return w, nil
|
2020-07-22 02:41:39 +00:00
|
|
|
}
|
|
|
|
|
2021-01-04 21:44:15 +00:00
|
|
|
func inputMnemonic(cliCtx *cli.Context) (mnemonicPhrase string, err error) {
|
2020-07-22 02:41:39 +00:00
|
|
|
if cliCtx.IsSet(flags.MnemonicFileFlag.Name) {
|
|
|
|
mnemonicFilePath := cliCtx.String(flags.MnemonicFileFlag.Name)
|
2022-04-18 20:42:07 +00:00
|
|
|
data, err := os.ReadFile(mnemonicFilePath) // #nosec G304 -- ReadFile is safe
|
2020-07-22 02:41:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
enteredMnemonic := string(data)
|
2021-04-05 20:42:03 +00:00
|
|
|
if err := ValidateMnemonic(enteredMnemonic); err != nil {
|
2020-07-22 02:41:39 +00:00
|
|
|
return "", errors.Wrap(err, "mnemonic phrase did not pass validation")
|
|
|
|
}
|
|
|
|
return enteredMnemonic, nil
|
|
|
|
}
|
2020-08-06 20:33:28 +00:00
|
|
|
allowedLanguages := map[string][]string{
|
|
|
|
"chinese_simplified": wordlists.ChineseSimplified,
|
|
|
|
"chinese_traditional": wordlists.ChineseTraditional,
|
2021-05-05 20:51:05 +00:00
|
|
|
"czech": wordlists.Czech,
|
|
|
|
"english": wordlists.English,
|
2020-08-06 20:33:28 +00:00
|
|
|
"french": wordlists.French,
|
|
|
|
"japanese": wordlists.Japanese,
|
|
|
|
"korean": wordlists.Korean,
|
2021-05-05 20:51:05 +00:00
|
|
|
"italian": wordlists.Italian,
|
2020-08-06 20:33:28 +00:00
|
|
|
"spanish": wordlists.Spanish,
|
|
|
|
}
|
|
|
|
languages := make([]string, 0)
|
|
|
|
for k := range allowedLanguages {
|
|
|
|
languages = append(languages, k)
|
|
|
|
}
|
|
|
|
sort.Strings(languages)
|
2021-09-17 21:55:24 +00:00
|
|
|
selectedLanguage, err := prompt.ValidatePrompt(
|
2020-08-20 17:53:09 +00:00
|
|
|
os.Stdin,
|
2020-08-06 20:33:28 +00:00
|
|
|
fmt.Sprintf("Enter the language of your seed phrase: %s", strings.Join(languages, ", ")),
|
|
|
|
func(input string) error {
|
|
|
|
if _, ok := allowedLanguages[input]; !ok {
|
|
|
|
return errors.New("input not in the list of allowed languages")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
)
|
|
|
|
if err != nil {
|
2020-10-14 17:39:52 +00:00
|
|
|
return "", fmt.Errorf("could not get mnemonic language: %w", err)
|
2020-08-06 20:33:28 +00:00
|
|
|
}
|
|
|
|
bip39.SetWordList(allowedLanguages[selectedLanguage])
|
2021-09-17 21:55:24 +00:00
|
|
|
mnemonicPhrase, err = prompt.ValidatePrompt(
|
2020-08-20 17:53:09 +00:00
|
|
|
os.Stdin,
|
|
|
|
"Enter the seed phrase for the wallet you would like to recover",
|
2021-04-05 20:42:03 +00:00
|
|
|
ValidateMnemonic)
|
2020-07-22 02:41:39 +00:00
|
|
|
if err != nil {
|
2020-10-14 17:39:52 +00:00
|
|
|
return "", fmt.Errorf("could not get mnemonic phrase: %w", err)
|
2020-07-22 02:41:39 +00:00
|
|
|
}
|
2020-07-29 04:55:26 +00:00
|
|
|
return mnemonicPhrase, nil
|
2020-07-22 02:41:39 +00:00
|
|
|
}
|
|
|
|
|
2020-07-27 14:03:30 +00:00
|
|
|
func inputNumAccounts(cliCtx *cli.Context) (int64, error) {
|
|
|
|
if cliCtx.IsSet(flags.NumAccountsFlag.Name) {
|
|
|
|
numAccounts := cliCtx.Int64(flags.NumAccountsFlag.Name)
|
2020-11-16 22:26:04 +00:00
|
|
|
if numAccounts <= 0 {
|
|
|
|
return 0, errors.New("must recover at least 1 account")
|
|
|
|
}
|
2020-07-27 14:03:30 +00:00
|
|
|
return numAccounts, nil
|
|
|
|
}
|
2021-09-17 21:55:24 +00:00
|
|
|
numAccounts, err := prompt.ValidatePrompt(os.Stdin, "Enter how many accounts you would like to generate from the mnemonic", prompt.ValidateNumber)
|
2020-07-27 14:03:30 +00:00
|
|
|
if err != nil {
|
2020-07-29 04:55:26 +00:00
|
|
|
return 0, err
|
2020-07-27 14:03:30 +00:00
|
|
|
}
|
|
|
|
numAccountsInt, err := strconv.Atoi(numAccounts)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2020-11-16 22:26:04 +00:00
|
|
|
if numAccountsInt <= 0 {
|
|
|
|
return 0, errors.New("must recover at least 1 account")
|
|
|
|
}
|
2020-07-27 14:03:30 +00:00
|
|
|
return int64(numAccountsInt), nil
|
|
|
|
}
|
|
|
|
|
2021-04-05 20:42:03 +00:00
|
|
|
// ValidateMnemonic ensures that it is not empty and that the count of the words are
|
|
|
|
// as specified(currently 24).
|
|
|
|
func ValidateMnemonic(mnemonic string) error {
|
2020-07-22 02:41:39 +00:00
|
|
|
if strings.Trim(mnemonic, " ") == "" {
|
2022-03-13 23:30:11 +00:00
|
|
|
return ErrEmptyMnemonic
|
2020-07-22 02:41:39 +00:00
|
|
|
}
|
|
|
|
words := strings.Split(mnemonic, " ")
|
2022-03-13 23:30:11 +00:00
|
|
|
validWordCount := 0
|
|
|
|
for _, word := range words {
|
2020-07-22 02:41:39 +00:00
|
|
|
if strings.Trim(word, " ") == "" {
|
2022-03-13 23:30:11 +00:00
|
|
|
continue
|
2020-07-22 02:41:39 +00:00
|
|
|
}
|
2022-03-13 23:30:11 +00:00
|
|
|
validWordCount += 1
|
2020-07-22 02:41:39 +00:00
|
|
|
}
|
2022-03-13 23:30:11 +00:00
|
|
|
if validWordCount != phraseWordCount {
|
|
|
|
return errors.Wrapf(ErrIncorrectWordNumber, "phrase must be %d words, entered %d", phraseWordCount, validWordCount)
|
2020-07-22 02:41:39 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|