2020-08-13 20:27:42 +00:00
|
|
|
package rpc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-09-22 14:49:07 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2020-09-24 16:47:13 +00:00
|
|
|
"strings"
|
2020-08-13 20:27:42 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/dgrijalva/jwt-go"
|
2020-09-03 15:11:17 +00:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/validator/accounts/v2"
|
2020-09-22 14:49:07 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/fileutil"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2020-09-03 15:11:17 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/promptutil"
|
2020-09-22 11:49:58 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/timeutils"
|
2020-09-17 01:34:42 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/validator/accounts/v2/wallet"
|
2020-09-22 14:49:07 +00:00
|
|
|
"golang.org/x/crypto/bcrypt"
|
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/status"
|
2020-08-13 20:27:42 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2020-10-10 02:07:28 +00:00
|
|
|
tokenExpiryLength = time.Hour
|
2020-08-13 20:27:42 +00:00
|
|
|
hashCost = 8
|
|
|
|
)
|
|
|
|
|
|
|
|
// Signup to authenticate access to the validator RPC API using bcrypt and
|
|
|
|
// a sufficiently strong password check.
|
|
|
|
func (s *Server) Signup(ctx context.Context, req *pb.AuthRequest) (*pb.AuthResponse, error) {
|
2020-10-10 02:07:28 +00:00
|
|
|
walletDir := s.walletDir
|
|
|
|
if strings.TrimSpace(req.WalletDir) != "" {
|
|
|
|
walletDir = req.WalletDir
|
|
|
|
}
|
2020-08-13 20:27:42 +00:00
|
|
|
// First, we check if the validator already has a password. In this case,
|
2020-10-10 02:07:28 +00:00
|
|
|
// the user should be logged in as normal.
|
|
|
|
if fileutil.FileExists(filepath.Join(walletDir, wallet.HashedPasswordFileName)) {
|
|
|
|
return s.Login(ctx, req)
|
2020-08-13 20:27:42 +00:00
|
|
|
}
|
|
|
|
// We check the strength of the password to ensure it is high-entropy,
|
|
|
|
// has the required character count, and contains only unicode characters.
|
|
|
|
if err := promptutil.ValidatePasswordInput(req.Password); err != nil {
|
2020-10-10 02:07:28 +00:00
|
|
|
return nil, status.Error(codes.InvalidArgument, "Could not validate wallet password input")
|
2020-08-13 20:27:42 +00:00
|
|
|
}
|
|
|
|
hashedPassword, err := bcrypt.GenerateFromPassword([]byte(req.Password), hashCost)
|
|
|
|
if err != nil {
|
2020-09-22 14:49:07 +00:00
|
|
|
return nil, errors.Wrap(err, "could not generate hashed password")
|
2020-08-13 20:27:42 +00:00
|
|
|
}
|
2020-10-10 02:07:28 +00:00
|
|
|
hashFilePath := filepath.Join(walletDir, wallet.HashedPasswordFileName)
|
2020-09-22 14:49:07 +00:00
|
|
|
// Write the config file to disk.
|
2020-10-10 02:07:28 +00:00
|
|
|
if err := os.MkdirAll(walletDir, os.ModePerm); err != nil {
|
2020-09-22 14:49:07 +00:00
|
|
|
return nil, status.Error(codes.Internal, err.Error())
|
2020-08-13 20:27:42 +00:00
|
|
|
}
|
2020-09-22 14:49:07 +00:00
|
|
|
if err := ioutil.WriteFile(hashFilePath, hashedPassword, params.BeaconIoConfig().ReadWritePermissions); err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "could not write hashed password for wallet to disk: %v", err)
|
2020-09-03 15:11:17 +00:00
|
|
|
}
|
2020-08-13 20:27:42 +00:00
|
|
|
return s.sendAuthResponse()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Login to authenticate with the validator RPC API using a password.
|
|
|
|
func (s *Server) Login(ctx context.Context, req *pb.AuthRequest) (*pb.AuthResponse, error) {
|
2020-10-10 02:07:28 +00:00
|
|
|
walletDir := s.walletDir
|
|
|
|
if strings.TrimSpace(req.WalletDir) != "" {
|
|
|
|
walletDir = req.WalletDir
|
|
|
|
}
|
|
|
|
// We check the strength of the password to ensure it is high-entropy,
|
|
|
|
// has the required character count, and contains only unicode characters.
|
|
|
|
if err := promptutil.ValidatePasswordInput(req.Password); err != nil {
|
|
|
|
return nil, status.Error(codes.InvalidArgument, "Could not validate wallet password input")
|
|
|
|
}
|
|
|
|
hashedPasswordPath := filepath.Join(walletDir, wallet.HashedPasswordFileName)
|
|
|
|
if !fileutil.FileExists(hashedPasswordPath) {
|
|
|
|
return nil, status.Error(codes.Internal, "Could not find hashed password on disk")
|
|
|
|
}
|
|
|
|
hashedPassword, err := fileutil.ReadFileAsBytes(hashedPasswordPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Error(codes.Internal, "Could not retrieve hashed password from disk")
|
|
|
|
}
|
|
|
|
// Compare the stored hashed password, with the hashed version of the password that was received.
|
|
|
|
if err := bcrypt.CompareHashAndPassword(hashedPassword, []byte(req.Password)); err != nil {
|
|
|
|
return nil, status.Error(codes.Unauthenticated, "Incorrect password")
|
2020-08-13 20:27:42 +00:00
|
|
|
}
|
2020-09-17 01:34:42 +00:00
|
|
|
if err := s.initializeWallet(ctx, &wallet.Config{
|
2020-10-10 02:07:28 +00:00
|
|
|
WalletDir: walletDir,
|
2020-09-03 15:11:17 +00:00
|
|
|
WalletPassword: req.Password,
|
|
|
|
}); err != nil {
|
2020-09-24 16:47:13 +00:00
|
|
|
if strings.Contains(err.Error(), "invalid checksum") {
|
|
|
|
return nil, status.Error(codes.Unauthenticated, "Incorrect password")
|
|
|
|
}
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not initialize wallet: %v", err)
|
2020-09-03 15:11:17 +00:00
|
|
|
}
|
2020-08-13 20:27:42 +00:00
|
|
|
return s.sendAuthResponse()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sends an auth response via gRPC containing a new JWT token.
|
|
|
|
func (s *Server) sendAuthResponse() (*pb.AuthResponse, error) {
|
|
|
|
// If everything is fine here, construct the auth token.
|
|
|
|
tokenString, expirationTime, err := s.createTokenString()
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Error(codes.Internal, "Could not create jwt token string")
|
|
|
|
}
|
|
|
|
return &pb.AuthResponse{
|
2020-08-14 16:30:11 +00:00
|
|
|
Token: tokenString,
|
2020-08-13 20:27:42 +00:00
|
|
|
TokenExpiration: expirationTime,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Creates a JWT token string using the JWT key with an expiration timestamp.
|
|
|
|
func (s *Server) createTokenString() (string, uint64, error) {
|
2020-08-14 16:30:11 +00:00
|
|
|
// Create a new token object, specifying signing method and the claims
|
|
|
|
// you would like it to contain.
|
2020-09-22 11:49:58 +00:00
|
|
|
expirationTime := timeutils.Now().Add(tokenExpiryLength)
|
2020-08-14 16:30:11 +00:00
|
|
|
token := jwt.NewWithClaims(jwt.SigningMethodHS256, jwt.StandardClaims{
|
2020-08-13 20:27:42 +00:00
|
|
|
ExpiresAt: expirationTime.Unix(),
|
2020-08-14 16:30:11 +00:00
|
|
|
})
|
|
|
|
// Sign and get the complete encoded token as a string using the secret
|
2020-08-13 20:27:42 +00:00
|
|
|
tokenString, err := token.SignedString(s.jwtKey)
|
|
|
|
if err != nil {
|
2020-08-14 16:30:11 +00:00
|
|
|
return "", 0, err
|
2020-08-13 20:27:42 +00:00
|
|
|
}
|
2020-08-14 16:30:11 +00:00
|
|
|
return tokenString, uint64(expirationTime.Unix()), nil
|
2020-08-13 20:27:42 +00:00
|
|
|
}
|
2020-09-03 15:11:17 +00:00
|
|
|
|
|
|
|
// Initialize a wallet and send it over a global feed.
|
2020-09-17 01:34:42 +00:00
|
|
|
func (s *Server) initializeWallet(ctx context.Context, cfg *wallet.Config) error {
|
2020-09-03 15:11:17 +00:00
|
|
|
// We first ensure the user has a wallet.
|
2020-09-30 14:13:37 +00:00
|
|
|
exists, err := wallet.Exists(cfg.WalletDir)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, wallet.CheckExistsErrMsg)
|
|
|
|
}
|
|
|
|
if !exists {
|
|
|
|
return wallet.ErrNoWalletFound
|
|
|
|
}
|
|
|
|
valid, err := wallet.IsValid(cfg.WalletDir)
|
|
|
|
if err == wallet.ErrNoWalletFound {
|
|
|
|
return wallet.ErrNoWalletFound
|
2020-09-03 15:11:17 +00:00
|
|
|
}
|
2020-09-30 14:13:37 +00:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, wallet.CheckValidityErrMsg)
|
|
|
|
}
|
|
|
|
if !valid {
|
|
|
|
return errors.New(wallet.InvalidWalletErrMsg)
|
|
|
|
}
|
|
|
|
|
2020-09-03 15:11:17 +00:00
|
|
|
// We fire an event with the opened wallet over
|
|
|
|
// a global feed signifying wallet initialization.
|
2020-09-22 14:49:07 +00:00
|
|
|
w, err := wallet.OpenWallet(ctx, &wallet.Config{
|
2020-09-03 15:11:17 +00:00
|
|
|
WalletDir: cfg.WalletDir,
|
|
|
|
WalletPassword: cfg.WalletPassword,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not open wallet")
|
|
|
|
}
|
2020-09-22 14:49:07 +00:00
|
|
|
|
2020-09-03 15:11:17 +00:00
|
|
|
s.walletInitialized = true
|
2020-09-22 14:49:07 +00:00
|
|
|
km, err := w.InitializeKeymanager(ctx, true /* skip mnemonic confirm */)
|
2020-09-04 00:58:36 +00:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not initialize keymanager")
|
|
|
|
}
|
2020-09-22 14:49:07 +00:00
|
|
|
s.keymanager = km
|
|
|
|
s.wallet = w
|
2020-10-10 02:07:28 +00:00
|
|
|
s.walletDir = cfg.WalletDir
|
|
|
|
|
|
|
|
// Only send over feed if we have validating keys.
|
|
|
|
validatingPublicKeys, err := km.FetchValidatingPublicKeys(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not check for validating public keys")
|
|
|
|
}
|
|
|
|
if len(validatingPublicKeys) > 0 {
|
|
|
|
s.walletInitializedFeed.Send(w)
|
|
|
|
}
|
2020-09-03 15:11:17 +00:00
|
|
|
return nil
|
|
|
|
}
|