2019-08-28 16:07:31 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/rand"
|
|
|
|
"encoding/json"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"log"
|
|
|
|
"os"
|
|
|
|
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/bls"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
numKeys = flag.Int("num-keys", 0, "Number of validator private/withdrawal keys to generate")
|
|
|
|
outputJSON = flag.String("output-json", "", "JSON file to write output to")
|
2019-08-30 04:32:08 +00:00
|
|
|
overwrite = flag.Bool("overwrite", false, "If the key file exists, it will be overwritten")
|
2019-08-28 16:07:31 +00:00
|
|
|
)
|
|
|
|
|
2019-09-09 21:31:19 +00:00
|
|
|
// UnencryptedKeysContainer defines the structure of the unecrypted key JSON file.
|
|
|
|
type UnencryptedKeysContainer struct {
|
|
|
|
Keys []*UnencryptedKeys `json:"keys"`
|
2019-08-28 16:07:31 +00:00
|
|
|
}
|
|
|
|
|
2019-09-09 21:31:19 +00:00
|
|
|
// UnencryptedKeys is the inner struct of the JSON file.
|
|
|
|
type UnencryptedKeys struct {
|
2019-08-28 16:07:31 +00:00
|
|
|
ValidatorKey []byte `json:"validator_key"`
|
|
|
|
WithdrawalKey []byte `json:"withdrawal_key"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
flag.Parse()
|
|
|
|
if *numKeys == 0 {
|
|
|
|
log.Fatal("Please specify --num-keys to generate")
|
|
|
|
}
|
|
|
|
if *outputJSON == "" {
|
|
|
|
log.Fatal("Please specify an --output-json file to write the unencrypted keys to")
|
|
|
|
}
|
|
|
|
|
2019-08-30 04:32:08 +00:00
|
|
|
if !*overwrite {
|
|
|
|
if _, err := os.Stat(*outputJSON); err == nil {
|
|
|
|
log.Fatal("The file exists. Use a different file name or the --overwrite flag")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-28 16:07:31 +00:00
|
|
|
file, err := os.Create(*outputJSON)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
defer func() {
|
|
|
|
if err := file.Close(); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
ctnr := generateUnencryptedKeys(rand.Reader)
|
2019-09-09 21:31:19 +00:00
|
|
|
if err := SaveUnencryptedKeysToFile(file, ctnr); err != nil {
|
2019-08-28 16:07:31 +00:00
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-09 21:31:19 +00:00
|
|
|
func generateUnencryptedKeys(r io.Reader) *UnencryptedKeysContainer {
|
|
|
|
ctnr := &UnencryptedKeysContainer{
|
|
|
|
Keys: make([]*UnencryptedKeys, *numKeys),
|
2019-08-28 16:07:31 +00:00
|
|
|
}
|
|
|
|
for i := 0; i < *numKeys; i++ {
|
|
|
|
signingKey, err := bls.RandKey(r)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
withdrawalKey, err := bls.RandKey(r)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
2019-09-09 21:31:19 +00:00
|
|
|
ctnr.Keys[i] = &UnencryptedKeys{
|
2019-08-28 16:07:31 +00:00
|
|
|
ValidatorKey: signingKey.Marshal(),
|
|
|
|
WithdrawalKey: withdrawalKey.Marshal(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ctnr
|
|
|
|
}
|
|
|
|
|
2019-09-09 21:31:19 +00:00
|
|
|
// SaveUnencryptedKeysToFile JSON encodes the container and writes to the writer.
|
|
|
|
func SaveUnencryptedKeysToFile(w io.Writer, ctnr *UnencryptedKeysContainer) error {
|
2019-08-28 16:07:31 +00:00
|
|
|
enc, err := json.Marshal(ctnr)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
n, err := w.Write(enc)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if n != len(enc) {
|
|
|
|
return fmt.Errorf("failed to write %d bytes to file, wrote %d", len(enc), n)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|