mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-19 00:04:12 +00:00
64f64f06bf
* removing flag requirement, can run web3signer without predefined public keys * placeholders for remote-keymanager-api * adding proto and accountschangedfeed * updating generated code * fix imports * fixing interface * adding work in progress apimiddleware code * started implementing functions for remote keymanager api * fixing generted code from proto * fixing protos * fixing import format * fixing proto generation again , didn't fix the first time * fixing imports again * continuing on implementing functions * implementing add function * implementing delete API function * handling errors for API * removing unusedcode and fixing format * fixing bazel * wip enable --web when running web3signer * fixing wallet check for web3signer * fixing apis * adding list remote keys unit test * import remote keys test * delete pubkeys tests * moving location of tests * adding unit tests * adding placeholder functions * adding more unit tests * fixing bazel * fixing build * fixing already slice issue with unit test * fixing linting * Update validator/client/validator.go Co-authored-by: Raul Jordan <raul@prysmaticlabs.com> * Update validator/keymanager/types.go Co-authored-by: Raul Jordan <raul@prysmaticlabs.com> * Update validator/node/node.go Co-authored-by: Raul Jordan <raul@prysmaticlabs.com> * Update validator/keymanager/types.go Co-authored-by: Raul Jordan <raul@prysmaticlabs.com> * Update validator/client/validator.go Co-authored-by: Raul Jordan <raul@prysmaticlabs.com> * adding comment on proto based on review * Update validator/keymanager/remote-web3signer/keymanager.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * Update validator/keymanager/remote-web3signer/keymanager.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * Update validator/rpc/standard_api.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * Update validator/rpc/standard_api.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * Update validator/rpc/standard_api.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * Update validator/rpc/standard_api.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * adding generated code based on review * updating based on feedback * fixing imports * fixing formatting * Update validator/rpc/standard_api.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * fixing event call * fixing dependency * updating bazel * Update validator/rpc/standard_api.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * Update validator/rpc/standard_api.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * Update validator/rpc/standard_api.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * Update validator/rpc/standard_api.go Co-authored-by: Radosław Kapka <rkapka@wp.pl> * addressing comment from review Co-authored-by: Raul Jordan <raul@prysmaticlabs.com> Co-authored-by: Radosław Kapka <rkapka@wp.pl>
385 lines
16 KiB
Go
385 lines
16 KiB
Go
package rpc
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
|
|
"github.com/golang/protobuf/ptypes/empty"
|
|
fieldparams "github.com/prysmaticlabs/prysm/config/fieldparams"
|
|
"github.com/prysmaticlabs/prysm/encoding/bytesutil"
|
|
ethpbservice "github.com/prysmaticlabs/prysm/proto/eth/service"
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager"
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager/derived"
|
|
slashingprotection "github.com/prysmaticlabs/prysm/validator/slashing-protection-history"
|
|
"github.com/prysmaticlabs/prysm/validator/slashing-protection-history/format"
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/status"
|
|
)
|
|
|
|
// ListKeystores implements the standard validator key management API.
|
|
func (s *Server) ListKeystores(
|
|
ctx context.Context, _ *empty.Empty,
|
|
) (*ethpbservice.ListKeystoresResponse, error) {
|
|
if !s.walletInitialized {
|
|
return nil, status.Error(codes.FailedPrecondition, "Prysm Wallet not initialized. Please create a new wallet.")
|
|
}
|
|
if s.validatorService == nil {
|
|
return nil, status.Error(codes.FailedPrecondition, "Validator service not ready. Please try again once validator is ready.")
|
|
}
|
|
km, err := s.validatorService.Keymanager()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get Prysm keymanager (possibly due to beacon node unavailable): %v", err)
|
|
}
|
|
if s.wallet.KeymanagerKind() != keymanager.Derived && s.wallet.KeymanagerKind() != keymanager.Local {
|
|
return nil, status.Errorf(codes.FailedPrecondition, "Prysm validator keys are not stored locally with this keymanager type.")
|
|
}
|
|
pubKeys, err := km.FetchValidatingPublicKeys(ctx)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not retrieve keystores: %v", err)
|
|
}
|
|
keystoreResponse := make([]*ethpbservice.ListKeystoresResponse_Keystore, len(pubKeys))
|
|
for i := 0; i < len(pubKeys); i++ {
|
|
keystoreResponse[i] = ðpbservice.ListKeystoresResponse_Keystore{
|
|
ValidatingPubkey: pubKeys[i][:],
|
|
}
|
|
if s.wallet.KeymanagerKind() == keymanager.Derived {
|
|
keystoreResponse[i].DerivationPath = fmt.Sprintf(derived.ValidatingKeyDerivationPathTemplate, i)
|
|
}
|
|
}
|
|
return ðpbservice.ListKeystoresResponse{
|
|
Data: keystoreResponse,
|
|
}, nil
|
|
}
|
|
|
|
// ImportKeystores allows for importing keystores into Prysm with their slashing protection history.
|
|
func (s *Server) ImportKeystores(
|
|
ctx context.Context, req *ethpbservice.ImportKeystoresRequest,
|
|
) (*ethpbservice.ImportKeystoresResponse, error) {
|
|
if !s.walletInitialized {
|
|
statuses := groupImportErrors(req, "Prysm Wallet not initialized. Please create a new wallet.")
|
|
return ðpbservice.ImportKeystoresResponse{Data: statuses}, nil
|
|
}
|
|
if s.validatorService == nil {
|
|
statuses := groupImportErrors(req, "Validator service not ready. Please try again once validator is ready.")
|
|
return ðpbservice.ImportKeystoresResponse{Data: statuses}, nil
|
|
}
|
|
km, err := s.validatorService.Keymanager()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get keymanager (possibly due to beacon node unavailable): %v", err)
|
|
}
|
|
importer, ok := km.(keymanager.Importer)
|
|
if !ok {
|
|
statuses := groupImportErrors(req, "Keymanager kind cannot import keys")
|
|
return ðpbservice.ImportKeystoresResponse{Data: statuses}, nil
|
|
}
|
|
if len(req.Keystores) == 0 {
|
|
return ðpbservice.ImportKeystoresResponse{}, nil
|
|
}
|
|
keystores := make([]*keymanager.Keystore, len(req.Keystores))
|
|
for i := 0; i < len(req.Keystores); i++ {
|
|
k := &keymanager.Keystore{}
|
|
err = json.Unmarshal([]byte(req.Keystores[i]), k)
|
|
if err != nil {
|
|
// we want to ignore unmarshal errors for now, proper status in importKeystore
|
|
k.Pubkey = "invalid format"
|
|
}
|
|
keystores[i] = k
|
|
}
|
|
if req.SlashingProtection != "" {
|
|
if err := slashingprotection.ImportStandardProtectionJSON(
|
|
ctx, s.valDB, bytes.NewBuffer([]byte(req.SlashingProtection)),
|
|
); err != nil {
|
|
statuses := make([]*ethpbservice.ImportedKeystoreStatus, len(req.Keystores))
|
|
for i := range statuses {
|
|
statuses[i] = ðpbservice.ImportedKeystoreStatus{
|
|
Status: ethpbservice.ImportedKeystoreStatus_ERROR,
|
|
Message: fmt.Sprintf("could not import slashing protection: %v", err),
|
|
}
|
|
}
|
|
return ðpbservice.ImportKeystoresResponse{Data: statuses}, nil
|
|
}
|
|
}
|
|
if len(req.Passwords) == 0 {
|
|
req.Passwords = make([]string, len(req.Keystores))
|
|
}
|
|
|
|
// req.Passwords and req.Keystores are checked for 0 length in code above.
|
|
if len(req.Passwords) > len(req.Keystores) {
|
|
req.Passwords = req.Passwords[:len(req.Keystores)]
|
|
}
|
|
if len(req.Passwords) < len(req.Keystores) {
|
|
passwordList := make([]string, len(req.Keystores))
|
|
copy(passwordList, req.Passwords)
|
|
req.Passwords = passwordList
|
|
}
|
|
|
|
statuses, err := importer.ImportKeystores(ctx, keystores, req.Passwords)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not import keystores: %v", err)
|
|
}
|
|
|
|
// If any of the keys imported had a slashing protection history before, we
|
|
// stop marking them as deleted from our validator database.
|
|
return ðpbservice.ImportKeystoresResponse{Data: statuses}, nil
|
|
}
|
|
|
|
func groupImportErrors(req *ethpbservice.ImportKeystoresRequest, errorMessage string) []*ethpbservice.ImportedKeystoreStatus {
|
|
statuses := make([]*ethpbservice.ImportedKeystoreStatus, len(req.Keystores))
|
|
for i := 0; i < len(req.Keystores); i++ {
|
|
statuses[i] = ðpbservice.ImportedKeystoreStatus{
|
|
Status: ethpbservice.ImportedKeystoreStatus_ERROR,
|
|
Message: errorMessage,
|
|
}
|
|
}
|
|
return statuses
|
|
}
|
|
|
|
// DeleteKeystores allows for deleting specified public keys from Prysm.
|
|
func (s *Server) DeleteKeystores(
|
|
ctx context.Context, req *ethpbservice.DeleteKeystoresRequest,
|
|
) (*ethpbservice.DeleteKeystoresResponse, error) {
|
|
if !s.walletInitialized {
|
|
statuses := groupExportErrors(req, "Prysm Wallet not initialized. Please create a new wallet.")
|
|
return ðpbservice.DeleteKeystoresResponse{Data: statuses}, nil
|
|
}
|
|
if s.validatorService == nil {
|
|
statuses := groupExportErrors(req, "Validator service not ready")
|
|
return ðpbservice.DeleteKeystoresResponse{Data: statuses}, nil
|
|
}
|
|
km, err := s.validatorService.Keymanager()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get keymanager (possibly due to beacon node unavailable): %v", err)
|
|
}
|
|
if len(req.Pubkeys) == 0 {
|
|
return ðpbservice.DeleteKeystoresResponse{Data: make([]*ethpbservice.DeletedKeystoreStatus, 0)}, nil
|
|
}
|
|
statuses, err := km.DeleteKeystores(ctx, req.Pubkeys)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not delete keys: %v", err)
|
|
}
|
|
|
|
statuses, err = s.transformDeletedKeysStatuses(ctx, req.Pubkeys, statuses)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not transform deleted keys statuses: %v", err)
|
|
}
|
|
|
|
exportedHistory, err := s.slashingProtectionHistoryForDeletedKeys(ctx, req.Pubkeys, statuses)
|
|
if err != nil {
|
|
log.Warnf("Could not get slashing protection history for deleted keys: %v", err)
|
|
statuses := groupExportErrors(req, "Non duplicate keys that were existing were deleted, but could not export slashing protection history.")
|
|
return ðpbservice.DeleteKeystoresResponse{Data: statuses}, nil
|
|
}
|
|
jsonHist, err := json.Marshal(exportedHistory)
|
|
if err != nil {
|
|
return nil, status.Errorf(
|
|
codes.Internal,
|
|
"Could not JSON marshal slashing protection history: %v",
|
|
err,
|
|
)
|
|
}
|
|
return ðpbservice.DeleteKeystoresResponse{
|
|
Data: statuses,
|
|
SlashingProtection: string(jsonHist),
|
|
}, nil
|
|
}
|
|
|
|
func groupExportErrors(req *ethpbservice.DeleteKeystoresRequest, errorMessage string) []*ethpbservice.DeletedKeystoreStatus {
|
|
statuses := make([]*ethpbservice.DeletedKeystoreStatus, len(req.Pubkeys))
|
|
for i := 0; i < len(req.Pubkeys); i++ {
|
|
statuses[i] = ðpbservice.DeletedKeystoreStatus{
|
|
Status: ethpbservice.DeletedKeystoreStatus_ERROR,
|
|
Message: errorMessage,
|
|
}
|
|
}
|
|
return statuses
|
|
}
|
|
|
|
// For a list of deleted keystore statuses, we check if any NOT_FOUND status actually
|
|
// has a corresponding public key in the database. In this case, we transform the status
|
|
// to NOT_ACTIVE, as we do have slashing protection history for it and should not mark it
|
|
// as NOT_FOUND when returning a response to the caller.
|
|
func (s *Server) transformDeletedKeysStatuses(
|
|
ctx context.Context, pubKeys [][]byte, statuses []*ethpbservice.DeletedKeystoreStatus,
|
|
) ([]*ethpbservice.DeletedKeystoreStatus, error) {
|
|
pubKeysInDB, err := s.publicKeysInDB(ctx)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get public keys from DB: %v", err)
|
|
}
|
|
if len(pubKeysInDB) > 0 {
|
|
for i := 0; i < len(pubKeys); i++ {
|
|
keyExistsInDB := pubKeysInDB[bytesutil.ToBytes48(pubKeys[i])]
|
|
if keyExistsInDB && statuses[i].Status == ethpbservice.DeletedKeystoreStatus_NOT_FOUND {
|
|
statuses[i].Status = ethpbservice.DeletedKeystoreStatus_NOT_ACTIVE
|
|
}
|
|
}
|
|
}
|
|
return statuses, nil
|
|
}
|
|
|
|
// Gets a map of all public keys in the database, useful for O(1) lookups.
|
|
func (s *Server) publicKeysInDB(ctx context.Context) (map[[fieldparams.BLSPubkeyLength]byte]bool, error) {
|
|
pubKeysInDB := make(map[[fieldparams.BLSPubkeyLength]byte]bool)
|
|
attestedPublicKeys, err := s.valDB.AttestedPublicKeys(ctx)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get attested public keys from DB: %v", err)
|
|
}
|
|
proposedPublicKeys, err := s.valDB.ProposedPublicKeys(ctx)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get proposed public keys from DB: %v", err)
|
|
}
|
|
for _, pk := range append(attestedPublicKeys, proposedPublicKeys...) {
|
|
pubKeysInDB[pk] = true
|
|
}
|
|
return pubKeysInDB, nil
|
|
}
|
|
|
|
// Exports slashing protection data for a list of DELETED or NOT_ACTIVE keys only to be used
|
|
// as part of the DeleteKeystores endpoint.
|
|
func (s *Server) slashingProtectionHistoryForDeletedKeys(
|
|
ctx context.Context, pubKeys [][]byte, statuses []*ethpbservice.DeletedKeystoreStatus,
|
|
) (*format.EIPSlashingProtectionFormat, error) {
|
|
// We select the keys that were DELETED or NOT_ACTIVE from the previous action
|
|
// and use that to filter our slashing protection export.
|
|
filteredKeys := make([][]byte, 0, len(pubKeys))
|
|
for i, pk := range pubKeys {
|
|
if statuses[i].Status == ethpbservice.DeletedKeystoreStatus_DELETED ||
|
|
statuses[i].Status == ethpbservice.DeletedKeystoreStatus_NOT_ACTIVE {
|
|
filteredKeys = append(filteredKeys, pk)
|
|
}
|
|
}
|
|
return slashingprotection.ExportStandardProtectionJSON(ctx, s.valDB, filteredKeys...)
|
|
}
|
|
|
|
// ListRemoteKeys returns a list of all public keys defined for web3signer keymanager type.
|
|
func (s *Server) ListRemoteKeys(ctx context.Context, _ *empty.Empty) (*ethpbservice.ListRemoteKeysResponse, error) {
|
|
if !s.walletInitialized {
|
|
return nil, status.Error(codes.FailedPrecondition, "Prysm Wallet not initialized. Please create a new wallet.")
|
|
}
|
|
if s.validatorService == nil {
|
|
return nil, status.Error(codes.FailedPrecondition, "Validator service not ready.")
|
|
}
|
|
km, err := s.validatorService.Keymanager()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get Prysm keymanager (possibly due to beacon node unavailable): %v", err)
|
|
}
|
|
if s.wallet.KeymanagerKind() != keymanager.Web3Signer {
|
|
return nil, status.Errorf(codes.FailedPrecondition, "Prysm Wallet is not of type Web3Signer. Please execute validator client with web3signer flags.")
|
|
}
|
|
pubKeys, err := km.FetchValidatingPublicKeys(ctx)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not retrieve keystores: %v", err)
|
|
}
|
|
keystoreResponse := make([]*ethpbservice.ListRemoteKeysResponse_Keystore, len(pubKeys))
|
|
for i := 0; i < len(pubKeys); i++ {
|
|
keystoreResponse[i] = ðpbservice.ListRemoteKeysResponse_Keystore{
|
|
Pubkey: pubKeys[i][:],
|
|
Url: s.validatorService.Web3SignerConfig.BaseEndpoint,
|
|
Readonly: true,
|
|
}
|
|
}
|
|
return ðpbservice.ListRemoteKeysResponse{
|
|
Data: keystoreResponse,
|
|
}, nil
|
|
}
|
|
|
|
// ImportRemoteKeys imports a list of public keys defined for web3signer keymanager type.
|
|
func (s *Server) ImportRemoteKeys(ctx context.Context, req *ethpbservice.ImportRemoteKeysRequest) (*ethpbservice.ImportRemoteKeysResponse, error) {
|
|
if !s.walletInitialized {
|
|
return nil, status.Error(codes.FailedPrecondition, "Prysm Wallet not initialized. Please create a new wallet.")
|
|
}
|
|
if s.validatorService == nil {
|
|
return nil, status.Error(codes.FailedPrecondition, "Validator service not ready.")
|
|
}
|
|
km, err := s.validatorService.Keymanager()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, fmt.Sprintf("Could not get Prysm keymanager (possibly due to beacon node unavailable): %v", err))
|
|
}
|
|
if s.wallet.KeymanagerKind() != keymanager.Web3Signer {
|
|
return nil, status.Errorf(codes.FailedPrecondition, "Prysm Wallet is not of type Web3Signer. Please execute validator client with web3signer flags.")
|
|
}
|
|
adder, ok := km.(keymanager.PublicKeyAdder)
|
|
if !ok {
|
|
statuses := groupImportRemoteKeysErrors(req, "Keymanager kind cannot import public keys for web3signer keymanager type.")
|
|
return ðpbservice.ImportRemoteKeysResponse{Data: statuses}, nil
|
|
}
|
|
|
|
remoteKeys := make([][fieldparams.BLSPubkeyLength]byte, len(req.RemoteKeys))
|
|
isUrlUsed := false
|
|
for i, obj := range req.RemoteKeys {
|
|
remoteKeys[i] = bytesutil.ToBytes48(obj.Pubkey)
|
|
if obj.Url != "" {
|
|
isUrlUsed = true
|
|
}
|
|
}
|
|
if isUrlUsed {
|
|
log.Warnf("Setting web3signer base url for imported keys is not supported. Prysm only uses the url from --validators-external-signer-url flag for web3signer.")
|
|
}
|
|
|
|
statuses, err := adder.AddPublicKeys(ctx, remoteKeys)
|
|
if err != nil {
|
|
sts := groupImportRemoteKeysErrors(req, fmt.Sprintf("Could not add keys;error: %v", err))
|
|
return ðpbservice.ImportRemoteKeysResponse{Data: sts}, nil
|
|
}
|
|
return ðpbservice.ImportRemoteKeysResponse{
|
|
Data: statuses,
|
|
}, nil
|
|
}
|
|
|
|
func groupImportRemoteKeysErrors(req *ethpbservice.ImportRemoteKeysRequest, errorMessage string) []*ethpbservice.ImportedRemoteKeysStatus {
|
|
statuses := make([]*ethpbservice.ImportedRemoteKeysStatus, len(req.RemoteKeys))
|
|
for i := 0; i < len(req.RemoteKeys); i++ {
|
|
statuses[i] = ðpbservice.ImportedRemoteKeysStatus{
|
|
Status: ethpbservice.ImportedRemoteKeysStatus_ERROR,
|
|
Message: errorMessage,
|
|
}
|
|
}
|
|
return statuses
|
|
}
|
|
|
|
// DeleteRemoteKeys deletes a list of public keys defined for web3signer keymanager type.
|
|
func (s *Server) DeleteRemoteKeys(ctx context.Context, req *ethpbservice.DeleteRemoteKeysRequest) (*ethpbservice.DeleteRemoteKeysResponse, error) {
|
|
if !s.walletInitialized {
|
|
return nil, status.Error(codes.FailedPrecondition, "Prysm Wallet not initialized. Please create a new wallet.")
|
|
}
|
|
if s.validatorService == nil {
|
|
return nil, status.Error(codes.FailedPrecondition, "Validator service not ready.")
|
|
}
|
|
km, err := s.validatorService.Keymanager()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get Prysm keymanager (possibly due to beacon node unavailable): %v", err)
|
|
}
|
|
if s.wallet.KeymanagerKind() != keymanager.Web3Signer {
|
|
return nil, status.Errorf(codes.FailedPrecondition, "Prysm Wallet is not of type Web3Signer. Please execute validator client with web3signer flags.")
|
|
}
|
|
deleter, ok := km.(keymanager.PublicKeyDeleter)
|
|
if !ok {
|
|
statuses := groupDeleteRemoteKeysErrors(req, "Keymanager kind cannot delete public keys for web3signer keymanager type.")
|
|
return ðpbservice.DeleteRemoteKeysResponse{Data: statuses}, nil
|
|
}
|
|
remoteKeys := make([][fieldparams.BLSPubkeyLength]byte, len(req.Pubkeys))
|
|
for i, key := range req.Pubkeys {
|
|
remoteKeys[i] = bytesutil.ToBytes48(key)
|
|
}
|
|
statuses, err := deleter.DeletePublicKeys(ctx, remoteKeys)
|
|
if err != nil {
|
|
sts := groupDeleteRemoteKeysErrors(req, fmt.Sprintf("Could not delete keys;error: %v", err))
|
|
return ðpbservice.DeleteRemoteKeysResponse{Data: sts}, nil
|
|
}
|
|
return ðpbservice.DeleteRemoteKeysResponse{
|
|
Data: statuses,
|
|
}, nil
|
|
}
|
|
|
|
func groupDeleteRemoteKeysErrors(req *ethpbservice.DeleteRemoteKeysRequest, errorMessage string) []*ethpbservice.DeletedRemoteKeysStatus {
|
|
statuses := make([]*ethpbservice.DeletedRemoteKeysStatus, len(req.Pubkeys))
|
|
for i := 0; i < len(req.Pubkeys); i++ {
|
|
statuses[i] = ðpbservice.DeletedRemoteKeysStatus{
|
|
Status: ethpbservice.DeletedRemoteKeysStatus_ERROR,
|
|
Message: errorMessage,
|
|
}
|
|
}
|
|
return statuses
|
|
}
|