package slashingprotection import ( "encoding/json" "fmt" "path/filepath" "github.com/pkg/errors" "github.com/prysmaticlabs/prysm/cmd/validator/flags" "github.com/prysmaticlabs/prysm/shared/cmd" "github.com/prysmaticlabs/prysm/shared/fileutil" "github.com/prysmaticlabs/prysm/validator/accounts/prompt" "github.com/prysmaticlabs/prysm/validator/db/kv" export "github.com/prysmaticlabs/prysm/validator/slashing-protection/local/standard-protection-format" "github.com/urfave/cli/v2" ) const ( jsonExportFileName = "slashing_protection.json" ) // ExportSlashingProtectionJSONCli extracts a validator's slashing protection // history from their database and formats it into an EIP-3076 standard JSON // file via a CLI entrypoint to make it easy to migrate machines or Ethereum consensus clients. // // Steps: // 1. Parse a path to the validator's datadir from the CLI context. // 2. Open the validator database. // 3. Call the function which actually exports the data from // from the validator's db into an EIP standard slashing protection format // 4. Format and save the JSON file to a user's specified output directory. func ExportSlashingProtectionJSONCli(cliCtx *cli.Context) error { log.Info( "This command exports your validator's attestation and proposal history into " + "a file that can then be imported into any other Prysm setup across computers", ) var err error dataDir := cliCtx.String(cmd.DataDirFlag.Name) if !cliCtx.IsSet(cmd.DataDirFlag.Name) { dataDir, err = prompt.InputDirectory(cliCtx, prompt.DataDirDirPromptText, cmd.DataDirFlag) if err != nil { return errors.Wrapf(err, "could not read directory value from input") } } // ensure that the validator.db is found under the specified dir or its subdirectories found, _, err := fileutil.RecursiveFileFind(kv.ProtectionDbFileName, dataDir) if err != nil { return errors.Wrapf(err, "error finding validator database at path %s", dataDir) } if !found { return fmt.Errorf( "validator.db file (validator database) was not found at path %s, so nothing to export", dataDir, ) } validatorDB, err := kv.NewKVStore(cliCtx.Context, dataDir, &kv.Config{}) if err != nil { return errors.Wrapf(err, "could not access validator database at path %s", dataDir) } defer func() { if err := validatorDB.Close(); err != nil { log.WithError(err).Errorf("Could not close validator DB") } }() eipJSON, err := export.ExportStandardProtectionJSON(cliCtx.Context, validatorDB) if err != nil { return errors.Wrap(err, "could not export slashing protection history") } outputDir, err := prompt.InputDirectory( cliCtx, "Enter your desired output directory for your slashing protection history file", flags.SlashingProtectionExportDirFlag, ) if err != nil { return errors.Wrap(err, "could not get slashing protection json file") } if outputDir == "" { return errors.New("output directory not specified") } exists, err := fileutil.HasDir(outputDir) if err != nil { return errors.Wrapf(err, "could not check if output directory %s already exists", outputDir) } if !exists { if err := fileutil.MkdirAll(outputDir); err != nil { return errors.Wrapf(err, "could not create output directory %s", outputDir) } } outputFilePath := filepath.Join(outputDir, jsonExportFileName) log.Infof("Writing slashing protection export JSON file to %s", outputFilePath) encoded, err := json.MarshalIndent(eipJSON, "", "\t") if err != nil { return errors.Wrap(err, "could not JSON marshal slashing protection history") } if err := fileutil.WriteFile(outputFilePath, encoded); err != nil { return errors.Wrapf(err, "could not write file to path %s", outputFilePath) } log.Infof( "Successfully wrote %s. You can import this file using Prysm's "+ "validator slashing-protection import command in another machine", outputFilePath, ) return nil }