2019-09-11 18:38:35 +00:00
|
|
|
package interop
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
|
|
|
"math/big"
|
2019-11-03 21:25:52 +00:00
|
|
|
"sync"
|
2019-09-11 18:38:35 +00:00
|
|
|
|
2022-08-16 12:20:13 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v3/encoding/bytesutil"
|
2022-01-11 18:42:03 +00:00
|
|
|
|
2019-09-11 18:38:35 +00:00
|
|
|
"github.com/pkg/errors"
|
2022-08-16 12:20:13 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v3/async"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/crypto/bls"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/crypto/hash"
|
2019-09-11 18:38:35 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
blsWithdrawalPrefixByte = byte(0)
|
|
|
|
)
|
|
|
|
|
|
|
|
// DeterministicallyGenerateKeys creates BLS private keys using a fixed curve order according to
|
2021-06-26 19:00:33 +00:00
|
|
|
// the algorithm specified in the Ethereum beacon chain specification interop mock start section found here:
|
2019-09-11 18:38:35 +00:00
|
|
|
// https://github.com/ethereum/eth2.0-pm/blob/a085c9870f3956d6228ed2a40cd37f0c6580ecd7/interop/mocked_start/README.md
|
2020-06-25 00:47:51 +00:00
|
|
|
func DeterministicallyGenerateKeys(startIndex, numKeys uint64) ([]bls.SecretKey, []bls.PublicKey, error) {
|
|
|
|
privKeys := make([]bls.SecretKey, numKeys)
|
|
|
|
pubKeys := make([]bls.PublicKey, numKeys)
|
2019-11-22 04:08:50 +00:00
|
|
|
type keys struct {
|
2020-06-25 00:47:51 +00:00
|
|
|
secrets []bls.SecretKey
|
|
|
|
publics []bls.PublicKey
|
2019-11-22 04:08:50 +00:00
|
|
|
}
|
2022-03-11 09:34:30 +00:00
|
|
|
// lint:ignore uintcast -- this is safe because we can reasonably expect that the number of keys is less than max int64.
|
2021-09-18 17:26:11 +00:00
|
|
|
results, err := async.Scatter(int(numKeys), func(offset int, entries int, _ *sync.RWMutex) (interface{}, error) {
|
2019-11-22 04:08:50 +00:00
|
|
|
secs, pubs, err := deterministicallyGenerateKeys(uint64(offset)+startIndex, uint64(entries))
|
|
|
|
return &keys{secrets: secs, publics: pubs}, err
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrap(err, "failed to generate keys")
|
|
|
|
}
|
|
|
|
for _, result := range results {
|
|
|
|
if keysExtent, ok := result.Extent.(*keys); ok {
|
|
|
|
copy(privKeys[result.Offset:], keysExtent.secrets)
|
|
|
|
copy(pubKeys[result.Offset:], keysExtent.publics)
|
|
|
|
} else {
|
|
|
|
return nil, nil, errors.New("extent not of expected type")
|
2019-11-03 21:25:52 +00:00
|
|
|
}
|
|
|
|
}
|
2019-11-22 04:08:50 +00:00
|
|
|
return privKeys, pubKeys, nil
|
2019-11-03 21:25:52 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 00:47:51 +00:00
|
|
|
func deterministicallyGenerateKeys(startIndex, numKeys uint64) ([]bls.SecretKey, []bls.PublicKey, error) {
|
|
|
|
privKeys := make([]bls.SecretKey, numKeys)
|
|
|
|
pubKeys := make([]bls.PublicKey, numKeys)
|
2019-09-11 18:38:35 +00:00
|
|
|
for i := startIndex; i < startIndex+numKeys; i++ {
|
|
|
|
enc := make([]byte, 32)
|
|
|
|
binary.LittleEndian.PutUint32(enc, uint32(i))
|
2022-06-27 13:34:38 +00:00
|
|
|
h := hash.Hash(enc)
|
2019-09-11 18:38:35 +00:00
|
|
|
// Reverse byte order to big endian for use with big ints.
|
2022-12-12 10:39:51 +00:00
|
|
|
num := bytesutil.LittleEndianBytesToBigInt(h[:])
|
2019-09-11 18:38:35 +00:00
|
|
|
order := new(big.Int)
|
|
|
|
var ok bool
|
|
|
|
order, ok = order.SetString(bls.CurveOrder, 10)
|
|
|
|
if !ok {
|
|
|
|
return nil, nil, errors.New("could not set bls curve order as big int")
|
|
|
|
}
|
|
|
|
num = num.Mod(num, order)
|
2019-09-14 14:46:38 +00:00
|
|
|
numBytes := num.Bytes()
|
|
|
|
// pad key at the start with zero bytes to make it into a 32 byte key
|
|
|
|
if len(numBytes) < 32 {
|
|
|
|
emptyBytes := make([]byte, 32-len(numBytes))
|
|
|
|
numBytes = append(emptyBytes, numBytes...)
|
|
|
|
}
|
|
|
|
priv, err := bls.SecretKeyFromBytes(numBytes)
|
2019-09-11 18:38:35 +00:00
|
|
|
if err != nil {
|
2019-09-14 14:46:38 +00:00
|
|
|
return nil, nil, errors.Wrapf(err, "could not create bls secret key at index %d from raw bytes", i)
|
2019-09-11 18:38:35 +00:00
|
|
|
}
|
|
|
|
privKeys[i-startIndex] = priv
|
|
|
|
pubKeys[i-startIndex] = priv.PublicKey()
|
|
|
|
}
|
|
|
|
return privKeys, pubKeys, nil
|
|
|
|
}
|