// +build linux,amd64 linux,arm64 // +build blst_enabled package blst import ( "fmt" "github.com/pkg/errors" "github.com/prysmaticlabs/prysm/shared/bls/common" "github.com/prysmaticlabs/prysm/shared/featureconfig" "github.com/prysmaticlabs/prysm/shared/params" "github.com/prysmaticlabs/prysm/shared/rand" blst "github.com/supranational/blst/bindings/go" ) // bls12SecretKey used in the BLS signature scheme. type bls12SecretKey struct { p *blst.SecretKey } // RandKey creates a new private key using a random method provided as an io.Reader. func RandKey() (common.SecretKey, error) { // Generate 32 bytes of randomness var ikm [32]byte _, err := rand.NewGenerator().Read(ikm[:]) if err != nil { return nil, err } // Defensive check, that we have not generated a secret key, secKey := &bls12SecretKey{blst.KeyGen(ikm[:])} if secKey.IsZero() { return nil, common.ErrZeroKey } return secKey, nil } // SecretKeyFromBytes creates a BLS private key from a BigEndian byte slice. func SecretKeyFromBytes(privKey []byte) (common.SecretKey, error) { if len(privKey) != params.BeaconConfig().BLSSecretKeyLength { return nil, fmt.Errorf("secret key must be %d bytes", params.BeaconConfig().BLSSecretKeyLength) } secKey := new(blst.SecretKey).Deserialize(privKey) if secKey == nil { return nil, errors.New("could not unmarshal bytes into secret key") } wrappedKey := &bls12SecretKey{p: secKey} if wrappedKey.IsZero() { return nil, common.ErrZeroKey } return wrappedKey, nil } // PublicKey obtains the public key corresponding to the BLS secret key. func (s *bls12SecretKey) PublicKey() common.PublicKey { return &PublicKey{p: new(blstPublicKey).From(s.p)} } // IsZero checks if the secret key is a zero key. func (s *bls12SecretKey) IsZero() bool { zeroKey := new(blst.SecretKey) return s.p.Equals(zeroKey) } // Sign a message using a secret key - in a beacon/validator client. // // In IETF draft BLS specification: // Sign(SK, message) -> signature: a signing algorithm that generates // a deterministic signature given a secret key SK and a message. // // In ETH2.0 specification: // def Sign(SK: int, message: Bytes) -> BLSSignature func (s *bls12SecretKey) Sign(msg []byte) common.Signature { if featureconfig.Get().SkipBLSVerify { return &Signature{} } signature := new(blstSignature).Sign(s.p, msg, dst) return &Signature{s: signature} } // Marshal a secret key into a LittleEndian byte slice. func (s *bls12SecretKey) Marshal() []byte { keyBytes := s.p.Serialize() if len(keyBytes) < params.BeaconConfig().BLSSecretKeyLength { emptyBytes := make([]byte, params.BeaconConfig().BLSSecretKeyLength-len(keyBytes)) keyBytes = append(emptyBytes, keyBytes...) } return keyBytes }