mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-12 04:30:04 +00:00
20f4d21b83
* Initial setup * Fix + Cleanup * Add query * Fix * Add epoch * James' review part 1 * James' review part 2 * James' review part 3 * Radek' review * Gazelle * Fix cycle * Start unit test * fixing part of the test * Mostly fix test * Fix tests * Cleanup * Handle error * Remove times * Fix all tests * Fix accidental deletion * Unmarshal epoch * Add custom_type * Small fix * Fix epoch * Lint fix * Add test + fix empty query panic * Add comment * Fix regex * Add correct error message * Change current epoch to use slot * Return error if incorrect epoch passed * Remove redundant type conversion * Fix tests * gaz * Remove nodeClient + pass slot * Remove slot from parameters * Fix tests * Fix test attempt 2 * Fix test attempt 2 * Remove nodeClient from ProposeExit * Fix * Fix tests --------- Co-authored-by: james-prysm <james@prysmaticlabs.com> Co-authored-by: james-prysm <90280386+james-prysm@users.noreply.github.com>
203 lines
6.6 KiB
Go
203 lines
6.6 KiB
Go
package accounts
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"path"
|
|
"strings"
|
|
|
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/blocks"
|
|
fieldparams "github.com/prysmaticlabs/prysm/v4/config/fieldparams"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/v4/io/file"
|
|
eth "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/v4/validator/client"
|
|
beacon_api "github.com/prysmaticlabs/prysm/v4/validator/client/beacon-api"
|
|
"github.com/prysmaticlabs/prysm/v4/validator/client/iface"
|
|
"github.com/prysmaticlabs/prysm/v4/validator/keymanager"
|
|
"google.golang.org/protobuf/types/known/emptypb"
|
|
)
|
|
|
|
// PerformExitCfg for account voluntary exits.
|
|
type PerformExitCfg struct {
|
|
ValidatorClient iface.ValidatorClient
|
|
NodeClient iface.NodeClient
|
|
Keymanager keymanager.IKeymanager
|
|
RawPubKeys [][]byte
|
|
FormattedPubKeys []string
|
|
OutputDirectory string
|
|
}
|
|
|
|
// ExitPassphrase exported for use in test.
|
|
const ExitPassphrase = "Exit my validator"
|
|
|
|
// Exit performs a voluntary exit on one or more accounts.
|
|
func (acm *AccountsCLIManager) Exit(ctx context.Context) error {
|
|
// User decided to cancel the voluntary exit.
|
|
if acm.rawPubKeys == nil && acm.formattedPubKeys == nil {
|
|
return nil
|
|
}
|
|
|
|
validatorClient, nodeClient, err := acm.prepareBeaconClients(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if nodeClient == nil {
|
|
return errors.New("could not prepare beacon node client")
|
|
}
|
|
syncStatus, err := (*nodeClient).GetSyncStatus(ctx, &emptypb.Empty{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if syncStatus == nil {
|
|
return errors.New("could not get sync status")
|
|
}
|
|
|
|
if syncStatus.Syncing {
|
|
return errors.New("could not perform exit: beacon node is syncing.")
|
|
}
|
|
|
|
cfg := PerformExitCfg{
|
|
*validatorClient,
|
|
*nodeClient,
|
|
acm.keymanager,
|
|
acm.rawPubKeys,
|
|
acm.formattedPubKeys,
|
|
acm.exitJSONOutputPath,
|
|
}
|
|
rawExitedKeys, trimmedExitedKeys, err := PerformVoluntaryExit(ctx, cfg)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
displayExitInfo(rawExitedKeys, trimmedExitedKeys)
|
|
|
|
return nil
|
|
}
|
|
|
|
// PerformVoluntaryExit uses gRPC clients to submit a voluntary exit message to a beacon node.
|
|
func PerformVoluntaryExit(
|
|
ctx context.Context, cfg PerformExitCfg,
|
|
) (rawExitedKeys [][]byte, formattedExitedKeys []string, err error) {
|
|
var rawNotExitedKeys [][]byte
|
|
genesisResponse, err := cfg.NodeClient.GetGenesis(ctx, &emptypb.Empty{})
|
|
if err != nil {
|
|
log.WithError(err).Errorf("voluntary exit failed: %v", err)
|
|
}
|
|
for i, key := range cfg.RawPubKeys {
|
|
// When output directory is present, only create the signed exit, but do not propose it.
|
|
// Otherwise, propose the exit immediately.
|
|
epoch, err := client.CurrentEpoch(genesisResponse.GenesisTime)
|
|
if err != nil {
|
|
log.WithError(err).Errorf("voluntary exit failed: %v", err)
|
|
}
|
|
if len(cfg.OutputDirectory) > 0 {
|
|
sve, err := client.CreateSignedVoluntaryExit(ctx, cfg.ValidatorClient, cfg.Keymanager.Sign, key, epoch)
|
|
if err != nil {
|
|
rawNotExitedKeys = append(rawNotExitedKeys, key)
|
|
msg := err.Error()
|
|
if strings.Contains(msg, blocks.ValidatorAlreadyExitedMsg) ||
|
|
strings.Contains(msg, blocks.ValidatorCannotExitYetMsg) {
|
|
log.Warningf("Could not create voluntary exit for account %s: %s", cfg.FormattedPubKeys[i], msg)
|
|
} else {
|
|
log.WithError(err).Errorf("voluntary exit failed for account %s", cfg.FormattedPubKeys[i])
|
|
}
|
|
} else if err := writeSignedVoluntaryExitJSON(ctx, sve, cfg.OutputDirectory); err != nil {
|
|
log.WithError(err).Error("failed to write voluntary exit")
|
|
}
|
|
} else if err := client.ProposeExit(ctx, cfg.ValidatorClient, cfg.Keymanager.Sign, key, epoch); err != nil {
|
|
rawNotExitedKeys = append(rawNotExitedKeys, key)
|
|
|
|
msg := err.Error()
|
|
if strings.Contains(msg, blocks.ValidatorAlreadyExitedMsg) ||
|
|
strings.Contains(msg, blocks.ValidatorCannotExitYetMsg) {
|
|
log.Warningf("Could not perform voluntary exit for account %s: %s", cfg.FormattedPubKeys[i], msg)
|
|
} else {
|
|
log.WithError(err).Errorf("voluntary exit failed for account %s", cfg.FormattedPubKeys[i])
|
|
}
|
|
}
|
|
}
|
|
|
|
rawExitedKeys = make([][]byte, 0)
|
|
formattedExitedKeys = make([]string, 0)
|
|
for i, key := range cfg.RawPubKeys {
|
|
found := false
|
|
for _, notExited := range rawNotExitedKeys {
|
|
if bytes.Equal(notExited, key) {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
if !found {
|
|
rawExitedKeys = append(rawExitedKeys, key)
|
|
formattedExitedKeys = append(formattedExitedKeys, cfg.FormattedPubKeys[i])
|
|
}
|
|
}
|
|
|
|
return rawExitedKeys, formattedExitedKeys, nil
|
|
}
|
|
|
|
func prepareAllKeys(validatingKeys [][fieldparams.BLSPubkeyLength]byte) (raw [][]byte, formatted []string) {
|
|
raw = make([][]byte, len(validatingKeys))
|
|
formatted = make([]string, len(validatingKeys))
|
|
for i, pk := range validatingKeys {
|
|
raw[i] = make([]byte, len(pk))
|
|
copy(raw[i], pk[:])
|
|
formatted[i] = fmt.Sprintf("%#x", bytesutil.Trunc(pk[:]))
|
|
}
|
|
return
|
|
}
|
|
|
|
func displayExitInfo(rawExitedKeys [][]byte, trimmedExitedKeys []string) {
|
|
if len(rawExitedKeys) > 0 {
|
|
urlFormattedPubKeys := make([]string, len(rawExitedKeys))
|
|
for i, key := range rawExitedKeys {
|
|
var baseUrl string
|
|
if params.BeaconConfig().ConfigName == params.PraterName || params.BeaconConfig().ConfigName == params.GoerliName {
|
|
baseUrl = "https://goerli.beaconcha.in/validator/"
|
|
} else {
|
|
baseUrl = "https://beaconcha.in/validator/"
|
|
}
|
|
// Remove '0x' prefix
|
|
urlFormattedPubKeys[i] = baseUrl + hexutil.Encode(key)[2:]
|
|
}
|
|
|
|
ifaceKeys := make([]interface{}, len(urlFormattedPubKeys))
|
|
for i, k := range urlFormattedPubKeys {
|
|
ifaceKeys[i] = k
|
|
}
|
|
|
|
info := fmt.Sprintf("Voluntary exit was successful for the accounts listed. "+
|
|
"URLs where you can track each validator's exit:\n"+strings.Repeat("%s\n", len(ifaceKeys)), ifaceKeys...)
|
|
|
|
log.WithField("publicKeys", strings.Join(trimmedExitedKeys, ", ")).Info(info)
|
|
} else {
|
|
log.Info("No successful voluntary exits")
|
|
}
|
|
}
|
|
|
|
func writeSignedVoluntaryExitJSON(ctx context.Context, sve *eth.SignedVoluntaryExit, outputDirectory string) error {
|
|
if err := file.MkdirAll(outputDirectory); err != nil {
|
|
return err
|
|
}
|
|
|
|
jsve := beacon_api.JsonifySignedVoluntaryExits([]*eth.SignedVoluntaryExit{sve})[0]
|
|
b, err := json.Marshal(jsve)
|
|
if err != nil {
|
|
return errors.Wrap(err, "failed to marshal JSON signed voluntary exit")
|
|
}
|
|
|
|
filepath := path.Join(outputDirectory, fmt.Sprintf("validator-exit-%s.json", jsve.Exit.ValidatorIndex))
|
|
if err := file.WriteFile(filepath, b); err != nil {
|
|
return errors.Wrap(err, "failed to write validator exist json")
|
|
}
|
|
|
|
log.Infof("Wrote signed validator exit JSON to %s", filepath)
|
|
|
|
return nil
|
|
}
|