Skip to content

Commit

Permalink
feat: enable more conversions between encoded and raw versions
Browse files Browse the repository at this point in the history
This allows to pass CA from config and save result back as PEM
certificates used e.g. to pass to some components.

Signed-off-by: Andrey Smirnov <smirnov.andrey@gmail.com>
  • Loading branch information
smira authored and talos-bot committed Jan 19, 2021
1 parent e0dd56a commit bda0e9c
Showing 1 changed file with 68 additions and 1 deletion.
69 changes: 68 additions & 1 deletion x509/x509.go
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,9 @@ type CertificateSigningRequest struct {
// KeyPair represents a certificate and key pair.
type KeyPair struct {
*tls.Certificate

CrtPEM []byte
KeyPEM []byte
}

// PEMEncodedCertificateAndKey represents a PEM encoded certificate and
Expand Down Expand Up @@ -230,6 +233,29 @@ func NewSelfSignedCertificateAuthority(setters ...Option) (ca *CertificateAuthor
return Ed25519CertificateAuthority(crt)
}

// NewCertificateAuthorityFromCertificateAndKey builds CertificateAuthority from PEMEncodedCertificateAndKey.
func NewCertificateAuthorityFromCertificateAndKey(p *PEMEncodedCertificateAndKey, setters ...Option) (ca *CertificateAuthority, err error) {
opts := NewDefaultOptions(setters...)

ca = &CertificateAuthority{
CrtPEM: p.Crt,
KeyPEM: p.Key,
}

ca.Crt, err = p.GetCert()
if err != nil {
return
}

if opts.RSA {
ca.Key, err = p.GetRSAKey()
} else {
ca.Key, err = p.GetEd25519Key()
}

return
}

// NewCertificateSigningRequest creates a CSR. If the IPAddresses or DNSNames options are not
// specified, the CSR will be generated with the default values set in
// NewDefaultOptions.
Expand Down Expand Up @@ -493,7 +519,9 @@ func NewKeyPair(ca *CertificateAuthority, setters ...Option) (keypair *KeyPair,
}

keypair = &KeyPair{
&x509KeyPair,
Certificate: &x509KeyPair,
CrtPEM: crt.X509CertificatePEM,
KeyPEM: identity.Key,
}

return keypair, nil
Expand Down Expand Up @@ -521,6 +549,24 @@ func NewCertificateAndKeyFromFiles(crt, key string) (p *PEMEncodedCertificateAnd
return p, nil
}

// NewCertificateAndKeyFromCertificateAuthority initializes and returns a
// PEMEncodedCertificateAndKey from the CertificateAuthority.
func NewCertificateAndKeyFromCertificateAuthority(ca *CertificateAuthority) *PEMEncodedCertificateAndKey {
return &PEMEncodedCertificateAndKey{
Crt: ca.CrtPEM,
Key: ca.KeyPEM,
}
}

// NewCertificateAndKeyFromKeyPair initializes and returns a
// PEMEncodedCertificateAndKey from the KeyPair.
func NewCertificateAndKeyFromKeyPair(keyPair *KeyPair) *PEMEncodedCertificateAndKey {
return &PEMEncodedCertificateAndKey{
Crt: keyPair.CrtPEM,
Key: keyPair.KeyPEM,
}
}

// NewEd25519CSRAndIdentity generates and PEM encoded certificate and key, along with a
// CSR for the generated key.
func NewEd25519CSRAndIdentity(dnsNames []string, ips []net.IP) (csr *CertificateSigningRequest, identity *PEMEncodedCertificateAndKey, err error) {
Expand Down Expand Up @@ -670,6 +716,27 @@ func (p *PEMEncodedCertificateAndKey) GetRSAKey() (*rsa.PrivateKey, error) {
return key, nil
}

// GetEd25519Key parses PEM-encoded Ed25519 key.
func (p *PEMEncodedCertificateAndKey) GetEd25519Key() (ed25519.PrivateKey, error) {
block, _ := pem.Decode(p.Key)
if block == nil {
return nil, fmt.Errorf("failed to parse PEM block")
}

key, err := x509.ParsePKCS8PrivateKey(block.Bytes)
if err != nil {
return nil, fmt.Errorf("failed to parse Ed25519 key: %w", err)
}

ed25519Key, ok := key.(ed25519.PrivateKey)

if !ok {
return nil, fmt.Errorf("failed parsing Ed25519 key, got wrong key type")
}

return ed25519Key, nil
}

// NewCertficateAndKey generates a new key and certificate signed by a CA.
//
//nolint: gocyclo
Expand Down

0 comments on commit bda0e9c

Please sign in to comment.