mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2024-12-25 21:07:18 +00:00
5fd03f8fb0
Co-authored-by: Preston Van Loon <preston@prysmaticlabs.com>
458 lines
14 KiB
Go
458 lines
14 KiB
Go
package wallet
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/shared/fileutil"
|
|
"github.com/prysmaticlabs/prysm/shared/promptutil"
|
|
"github.com/prysmaticlabs/prysm/validator/accounts/prompt"
|
|
"github.com/prysmaticlabs/prysm/validator/flags"
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager"
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager/derived"
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager/imported"
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager/remote"
|
|
"github.com/sirupsen/logrus"
|
|
"github.com/urfave/cli/v2"
|
|
)
|
|
|
|
const (
|
|
// KeymanagerConfigFileName for the keymanager used by the wallet: imported, derived, or remote.
|
|
KeymanagerConfigFileName = "keymanageropts.json"
|
|
// NewWalletPasswordPromptText for wallet creation.
|
|
NewWalletPasswordPromptText = "New wallet password"
|
|
// WalletPasswordPromptText for wallet unlocking.
|
|
WalletPasswordPromptText = "Wallet password"
|
|
// ConfirmPasswordPromptText for confirming a wallet password.
|
|
ConfirmPasswordPromptText = "Confirm password"
|
|
// DefaultWalletPasswordFile used to store a wallet password with appropriate permissions
|
|
// if a user signs up via the Prysm web UI via RPC.
|
|
DefaultWalletPasswordFile = "walletpassword.txt"
|
|
// CheckExistsErrMsg for when there is an error while checking for a wallet
|
|
CheckExistsErrMsg = "could not check if wallet exists"
|
|
// CheckValidityErrMsg for when there is an error while checking wallet validity
|
|
CheckValidityErrMsg = "could not check if wallet is valid"
|
|
// InvalidWalletErrMsg for when a directory does not contain a valid wallet
|
|
InvalidWalletErrMsg = "directory does not contain valid wallet"
|
|
)
|
|
|
|
var (
|
|
// ErrNoWalletFound signifies there was no wallet directory found on-disk.
|
|
ErrNoWalletFound = errors.New(
|
|
"no wallet found. You can create a new wallet with `validator wallet create`. " +
|
|
"If you already did, perhaps you created a wallet in a custom directory, which you can specify using " +
|
|
"`--wallet-dir=/path/to/my/wallet`",
|
|
)
|
|
// KeymanagerKindSelections as friendly text.
|
|
KeymanagerKindSelections = map[keymanager.Kind]string{
|
|
keymanager.Imported: "Imported Wallet (Recommended)",
|
|
keymanager.Derived: "HD Wallet",
|
|
keymanager.Remote: "Remote Signing Wallet (Advanced)",
|
|
}
|
|
// ValidateExistingPass checks that an input cannot be empty.
|
|
ValidateExistingPass = func(input string) error {
|
|
if input == "" {
|
|
return errors.New("password input cannot be empty")
|
|
}
|
|
return nil
|
|
}
|
|
)
|
|
|
|
// Config to open a wallet programmatically.
|
|
type Config struct {
|
|
WalletDir string
|
|
KeymanagerKind keymanager.Kind
|
|
WalletPassword string
|
|
}
|
|
|
|
// Wallet is a primitive in Prysm's account management which
|
|
// has the capability of creating new accounts, reading existing accounts,
|
|
// and providing secure access to eth2 secrets depending on an
|
|
// associated keymanager (either imported, derived, or remote signing enabled).
|
|
type Wallet struct {
|
|
walletDir string
|
|
accountsPath string
|
|
configFilePath string
|
|
walletPassword string
|
|
keymanagerKind keymanager.Kind
|
|
}
|
|
|
|
// New creates a struct from config values.
|
|
func New(cfg *Config) *Wallet {
|
|
accountsPath := filepath.Join(cfg.WalletDir, cfg.KeymanagerKind.String())
|
|
return &Wallet{
|
|
walletDir: cfg.WalletDir,
|
|
accountsPath: accountsPath,
|
|
keymanagerKind: cfg.KeymanagerKind,
|
|
walletPassword: cfg.WalletPassword,
|
|
}
|
|
}
|
|
|
|
// Exists checks if directory at walletDir exists
|
|
func Exists(walletDir string) (bool, error) {
|
|
dirExists, err := fileutil.HasDir(walletDir)
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "could not parse wallet directory")
|
|
}
|
|
isValid, err := IsValid(walletDir)
|
|
if errors.Is(err, ErrNoWalletFound) {
|
|
return false, nil
|
|
} else if err != nil {
|
|
return false, errors.Wrap(err, "could not check if dir is valid")
|
|
}
|
|
return dirExists && isValid, nil
|
|
}
|
|
|
|
// IsValid checks if a folder contains a single key directory such as `derived`, `remote` or `imported`.
|
|
// Returns true if one of those subdirectories exist, false otherwise.
|
|
func IsValid(walletDir string) (bool, error) {
|
|
expanded, err := fileutil.ExpandPath(walletDir)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
f, err := os.Open(expanded)
|
|
if err != nil {
|
|
if strings.Contains(err.Error(), "no such file") ||
|
|
strings.Contains(err.Error(), "cannot find the file") ||
|
|
strings.Contains(err.Error(), "cannot find the path") {
|
|
return false, nil
|
|
}
|
|
return false, err
|
|
}
|
|
defer func() {
|
|
if err := f.Close(); err != nil {
|
|
log.Debugf("Could not close directory: %s", expanded)
|
|
}
|
|
}()
|
|
names, err := f.Readdirnames(-1)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if len(names) == 0 {
|
|
return false, ErrNoWalletFound
|
|
}
|
|
|
|
// Count how many wallet types we have in the directory
|
|
numWalletTypes := 0
|
|
for _, name := range names {
|
|
// Nil error means input name is `derived`, `remote` or `imported`
|
|
_, err = keymanager.ParseKind(name)
|
|
if err == nil {
|
|
numWalletTypes++
|
|
}
|
|
}
|
|
return numWalletTypes == 1, nil
|
|
}
|
|
|
|
// OpenWalletOrElseCli tries to open the wallet and if it fails or no wallet
|
|
// is found, invokes a callback function.
|
|
func OpenWalletOrElseCli(cliCtx *cli.Context, otherwise func(cliCtx *cli.Context) (*Wallet, error)) (*Wallet, error) {
|
|
exists, err := Exists(cliCtx.String(flags.WalletDirFlag.Name))
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, CheckExistsErrMsg)
|
|
}
|
|
if !exists {
|
|
return otherwise(cliCtx)
|
|
}
|
|
isValid, err := IsValid(cliCtx.String(flags.WalletDirFlag.Name))
|
|
if errors.Is(err, ErrNoWalletFound) {
|
|
return otherwise(cliCtx)
|
|
}
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, CheckValidityErrMsg)
|
|
}
|
|
if !isValid {
|
|
return nil, errors.New(InvalidWalletErrMsg)
|
|
}
|
|
|
|
walletDir, err := prompt.InputDirectory(cliCtx, prompt.WalletDirPromptText, flags.WalletDirFlag)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
walletPassword, err := inputPassword(
|
|
cliCtx,
|
|
flags.WalletPasswordFileFlag,
|
|
WalletPasswordPromptText,
|
|
false, /* Do not confirm password */
|
|
ValidateExistingPass,
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return OpenWallet(cliCtx.Context, &Config{
|
|
WalletDir: walletDir,
|
|
WalletPassword: walletPassword,
|
|
})
|
|
}
|
|
|
|
// OpenWallet instantiates a wallet from a specified path. It checks the
|
|
// type of keymanager associated with the wallet by reading files in the wallet
|
|
// path, if applicable. If a wallet does not exist, returns an appropriate error.
|
|
func OpenWallet(_ context.Context, cfg *Config) (*Wallet, error) {
|
|
exists, err := Exists(cfg.WalletDir)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, CheckExistsErrMsg)
|
|
}
|
|
if !exists {
|
|
return nil, ErrNoWalletFound
|
|
}
|
|
valid, err := IsValid(cfg.WalletDir)
|
|
// ErrNoWalletFound represents both a directory that does not exist as well as an empty directory
|
|
if errors.Is(err, ErrNoWalletFound) {
|
|
return nil, ErrNoWalletFound
|
|
}
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, CheckValidityErrMsg)
|
|
}
|
|
if !valid {
|
|
return nil, errors.New(InvalidWalletErrMsg)
|
|
}
|
|
|
|
keymanagerKind, err := readKeymanagerKindFromWalletPath(cfg.WalletDir)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not read keymanager kind for wallet")
|
|
}
|
|
accountsPath := filepath.Join(cfg.WalletDir, keymanagerKind.String())
|
|
return &Wallet{
|
|
walletDir: cfg.WalletDir,
|
|
accountsPath: accountsPath,
|
|
keymanagerKind: keymanagerKind,
|
|
walletPassword: cfg.WalletPassword,
|
|
}, nil
|
|
}
|
|
|
|
// SaveWallet persists the wallet's directories to disk.
|
|
func (w *Wallet) SaveWallet() error {
|
|
if err := fileutil.MkdirAll(w.accountsPath); err != nil {
|
|
return errors.Wrap(err, "could not create wallet directory")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// KeymanagerKind used by the wallet.
|
|
func (w *Wallet) KeymanagerKind() keymanager.Kind {
|
|
return w.keymanagerKind
|
|
}
|
|
|
|
// AccountsDir for the wallet.
|
|
func (w *Wallet) AccountsDir() string {
|
|
return w.accountsPath
|
|
}
|
|
|
|
// Password for the wallet.
|
|
func (w *Wallet) Password() string {
|
|
return w.walletPassword
|
|
}
|
|
|
|
// InitializeKeymanager reads a keymanager config from disk at the wallet path,
|
|
// unmarshals it based on the wallet's keymanager kind, and returns its value.
|
|
func (w *Wallet) InitializeKeymanager(ctx context.Context) (keymanager.IKeymanager, error) {
|
|
var km keymanager.IKeymanager
|
|
var err error
|
|
switch w.KeymanagerKind() {
|
|
case keymanager.Imported:
|
|
km, err = imported.NewKeymanager(ctx, &imported.SetupConfig{
|
|
Wallet: w,
|
|
})
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not initialize imported keymanager")
|
|
}
|
|
case keymanager.Derived:
|
|
km, err = derived.NewKeymanager(ctx, &derived.SetupConfig{
|
|
Wallet: w,
|
|
})
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not initialize derived keymanager")
|
|
}
|
|
case keymanager.Remote:
|
|
configFile, err := w.ReadKeymanagerConfigFromDisk(ctx)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not read keymanager config")
|
|
}
|
|
opts, err := remote.UnmarshalOptionsFile(configFile)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not unmarshal keymanager config file")
|
|
}
|
|
km, err = remote.NewKeymanager(ctx, &remote.SetupConfig{
|
|
Opts: opts,
|
|
MaxMessageSize: 100000000,
|
|
})
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not initialize remote keymanager")
|
|
}
|
|
default:
|
|
return nil, fmt.Errorf("keymanager kind not supported: %s", w.keymanagerKind)
|
|
}
|
|
return km, nil
|
|
}
|
|
|
|
// WriteFileAtPath within the wallet directory given the desired path, filename, and raw data.
|
|
func (w *Wallet) WriteFileAtPath(_ context.Context, filePath, fileName string, data []byte) error {
|
|
accountPath := filepath.Join(w.accountsPath, filePath)
|
|
hasDir, err := fileutil.HasDir(accountPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !hasDir {
|
|
if err := fileutil.MkdirAll(accountPath); err != nil {
|
|
return errors.Wrapf(err, "could not create path: %s", accountPath)
|
|
}
|
|
}
|
|
fullPath := filepath.Join(accountPath, fileName)
|
|
if err := fileutil.WriteFile(fullPath, data); err != nil {
|
|
return errors.Wrapf(err, "could not write %s", filePath)
|
|
}
|
|
log.WithFields(logrus.Fields{
|
|
"path": fullPath,
|
|
"fileName": fileName,
|
|
}).Debug("Wrote new file at path")
|
|
return nil
|
|
}
|
|
|
|
// ReadFileAtPath within the wallet directory given the desired path and filename.
|
|
func (w *Wallet) ReadFileAtPath(_ context.Context, filePath, fileName string) ([]byte, error) {
|
|
accountPath := filepath.Join(w.accountsPath, filePath)
|
|
hasDir, err := fileutil.HasDir(accountPath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if !hasDir {
|
|
if err := fileutil.MkdirAll(accountPath); err != nil {
|
|
return nil, errors.Wrapf(err, "could not create path: %s", accountPath)
|
|
}
|
|
}
|
|
fullPath := filepath.Join(accountPath, fileName)
|
|
matches, err := filepath.Glob(fullPath)
|
|
if err != nil {
|
|
return []byte{}, errors.Wrap(err, "could not find file")
|
|
}
|
|
if len(matches) == 0 {
|
|
return []byte{}, fmt.Errorf("no files found in path: %s", fullPath)
|
|
}
|
|
rawData, err := ioutil.ReadFile(matches[0])
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not read path: %s", filePath)
|
|
}
|
|
return rawData, nil
|
|
}
|
|
|
|
// FileNameAtPath return the full file name for the requested file. It allows for finding the file
|
|
// with a regex pattern.
|
|
func (w *Wallet) FileNameAtPath(_ context.Context, filePath, fileName string) (string, error) {
|
|
accountPath := filepath.Join(w.accountsPath, filePath)
|
|
if err := fileutil.MkdirAll(accountPath); err != nil {
|
|
return "", errors.Wrapf(err, "could not create path: %s", accountPath)
|
|
}
|
|
fullPath := filepath.Join(accountPath, fileName)
|
|
matches, err := filepath.Glob(fullPath)
|
|
if err != nil {
|
|
return "", errors.Wrap(err, "could not find file")
|
|
}
|
|
if len(matches) == 0 {
|
|
return "", fmt.Errorf("no files found in path: %s", fullPath)
|
|
}
|
|
fullFileName := filepath.Base(matches[0])
|
|
return fullFileName, nil
|
|
}
|
|
|
|
// ReadKeymanagerConfigFromDisk opens a keymanager config file
|
|
// for reading if it exists at the wallet path.
|
|
func (w *Wallet) ReadKeymanagerConfigFromDisk(_ context.Context) (io.ReadCloser, error) {
|
|
configFilePath := filepath.Join(w.accountsPath, KeymanagerConfigFileName)
|
|
if !fileutil.FileExists(configFilePath) {
|
|
return nil, fmt.Errorf("no keymanager config file found at path: %s", w.accountsPath)
|
|
}
|
|
w.configFilePath = configFilePath
|
|
return os.Open(configFilePath)
|
|
|
|
}
|
|
|
|
// WriteKeymanagerConfigToDisk takes an encoded keymanager config file
|
|
// and writes it to the wallet path.
|
|
func (w *Wallet) WriteKeymanagerConfigToDisk(_ context.Context, encoded []byte) error {
|
|
configFilePath := filepath.Join(w.accountsPath, KeymanagerConfigFileName)
|
|
// Write the config file to disk.
|
|
if err := fileutil.WriteFile(configFilePath, encoded); err != nil {
|
|
return errors.Wrapf(err, "could not write config to path: %s", configFilePath)
|
|
}
|
|
log.WithField("configFilePath", configFilePath).Debug("Wrote keymanager config file to disk")
|
|
return nil
|
|
}
|
|
|
|
func readKeymanagerKindFromWalletPath(walletPath string) (keymanager.Kind, error) {
|
|
walletItem, err := os.Open(walletPath)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
defer func() {
|
|
if err := walletItem.Close(); err != nil {
|
|
log.WithField(
|
|
"path", walletPath,
|
|
).Errorf("Could not close wallet directory: %v", err)
|
|
}
|
|
}()
|
|
list, err := walletItem.Readdirnames(0) // 0 to read all files and folders.
|
|
if err != nil {
|
|
return 0, fmt.Errorf("could not read files in directory: %s", walletPath)
|
|
}
|
|
for _, n := range list {
|
|
keymanagerKind, err := keymanager.ParseKind(n)
|
|
if err == nil {
|
|
return keymanagerKind, nil
|
|
}
|
|
}
|
|
return 0, errors.New("no keymanager folder (imported, remote, derived) found in wallet path")
|
|
}
|
|
|
|
func inputPassword(
|
|
cliCtx *cli.Context,
|
|
passwordFileFlag *cli.StringFlag,
|
|
promptText string,
|
|
confirmPassword bool,
|
|
passwordValidator func(input string) error,
|
|
) (string, error) {
|
|
if cliCtx.IsSet(passwordFileFlag.Name) {
|
|
passwordFilePathInput := cliCtx.String(passwordFileFlag.Name)
|
|
data, err := fileutil.ReadFileAsBytes(passwordFilePathInput)
|
|
if err != nil {
|
|
return "", errors.Wrap(err, "could not read file as bytes")
|
|
}
|
|
enteredPassword := strings.TrimRight(string(data), "\r\n")
|
|
if err := passwordValidator(enteredPassword); err != nil {
|
|
return "", errors.Wrap(err, "password did not pass validation")
|
|
}
|
|
return enteredPassword, nil
|
|
}
|
|
var hasValidPassword bool
|
|
var walletPassword string
|
|
var err error
|
|
for !hasValidPassword {
|
|
walletPassword, err = promptutil.PasswordPrompt(promptText, passwordValidator)
|
|
if err != nil {
|
|
return "", fmt.Errorf("could not read account password: %w", err)
|
|
}
|
|
|
|
if confirmPassword {
|
|
passwordConfirmation, err := promptutil.PasswordPrompt(ConfirmPasswordPromptText, passwordValidator)
|
|
if err != nil {
|
|
return "", fmt.Errorf("could not read password confirmation: %w", err)
|
|
}
|
|
if walletPassword != passwordConfirmation {
|
|
log.Error("Passwords do not match")
|
|
continue
|
|
}
|
|
hasValidPassword = true
|
|
} else {
|
|
return walletPassword, nil
|
|
}
|
|
}
|
|
return walletPassword, nil
|
|
}
|