Skip to content

Commit

Permalink
fix: remove KeyEncipherment usage for non-RSA certificates (#48231)
Browse files Browse the repository at this point in the history
  • Loading branch information
nklaassen authored Oct 31, 2024
1 parent 637af67 commit 20e6a71
Show file tree
Hide file tree
Showing 4 changed files with 80 additions and 10 deletions.
8 changes: 7 additions & 1 deletion lib/tlsca/ca.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ package tlsca
import (
"crypto"
"crypto/rand"
"crypto/rsa"
"crypto/tls"
"crypto/x509"
"crypto/x509/pkix"
Expand Down Expand Up @@ -1199,7 +1200,12 @@ func (c *CertificateRequest) CheckAndSetDefaults() error {
return trace.BadParameter("missing parameter NotAfter")
}
if c.KeyUsage == 0 {
c.KeyUsage = x509.KeyUsageKeyEncipherment | x509.KeyUsageDigitalSignature
c.KeyUsage = x509.KeyUsageDigitalSignature
if _, isRSA := c.PublicKey.(*rsa.PublicKey); isRSA {
// The KeyEncipherment bit is necessary for RSA key exchanges
// https://datatracker.ietf.org/doc/html/rfc5280#section-4.2.1.3
c.KeyUsage |= x509.KeyUsageKeyEncipherment
}
}

c.DNSNames = utils.Deduplicate(c.DNSNames)
Expand Down
57 changes: 57 additions & 0 deletions lib/tlsca/ca_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,9 @@
package tlsca

import (
"crypto"
"crypto/tls"
"crypto/x509"
"crypto/x509/pkix"
"encoding/asn1"
"testing"
Expand All @@ -37,6 +39,7 @@ import (
apievents "github.com/gravitational/teleport/api/types/events"
"github.com/gravitational/teleport/api/utils/keys"
"github.com/gravitational/teleport/lib/cryptosuites"
"github.com/gravitational/teleport/lib/defaults"
"github.com/gravitational/teleport/lib/fixtures"
)

Expand Down Expand Up @@ -471,3 +474,57 @@ func TestIdentity_GetUserMetadata(t *testing.T) {
})
}
}

// TestKeyUsage asserts that only certs with RSA subject keys get the
// KeyEncipherment keyUsage extension.
func TestKeyUsage(t *testing.T) {
rsaKey, err := keys.ParsePrivateKey(fixtures.PEMBytes["rsa"])
require.NoError(t, err)
ecdsaKey, err := cryptosuites.GenerateKeyWithAlgorithm(cryptosuites.ECDSAP256)
require.NoError(t, err)
for _, tc := range []struct {
algo string
key crypto.Signer
expectCAKeyUsage x509.KeyUsage
expectSubjectKeyUsage x509.KeyUsage
}{
{
algo: "RSA",
key: rsaKey,
expectCAKeyUsage: x509.KeyUsageDigitalSignature | x509.KeyUsageCertSign | x509.KeyUsageCRLSign | x509.KeyUsageKeyEncipherment,
expectSubjectKeyUsage: x509.KeyUsageDigitalSignature | x509.KeyUsageKeyEncipherment,
},
{
algo: "ECDSA",
key: ecdsaKey,
expectCAKeyUsage: x509.KeyUsageDigitalSignature | x509.KeyUsageCertSign | x509.KeyUsageCRLSign,
expectSubjectKeyUsage: x509.KeyUsageDigitalSignature,
},
} {
t.Run(tc.algo, func(t *testing.T) {
caPEM, err := GenerateSelfSignedCAWithSigner(tc.key, pkix.Name{
CommonName: "teleport.example.com",
Organization: []string{"teleport.example.com"},
}, nil /*dnsNames*/, defaults.CATTL)
require.NoError(t, err)

ca, err := FromCertAndSigner(caPEM, tc.key)
require.NoError(t, err)
require.Equal(t, tc.expectCAKeyUsage, ca.Cert.KeyUsage)

subjectPEM, err := ca.GenerateCertificate(CertificateRequest{
PublicKey: tc.key.Public(),
Subject: pkix.Name{
CommonName: "teleport.example.com",
Organization: []string{"teleport.example.com"},
},
NotAfter: time.Now().Add(time.Hour),
})
require.NoError(t, err)

subject, err := ParseCertificatePEM(subjectPEM)
require.NoError(t, err)
require.Equal(t, tc.expectSubjectKeyUsage, subject.KeyUsage)
})
}
}
23 changes: 15 additions & 8 deletions lib/tlsca/parsegen.go
Original file line number Diff line number Diff line change
Expand Up @@ -92,15 +92,22 @@ func GenerateSelfSignedCAWithConfig(config GenerateCAConfig) (certPEM []byte, er
// signed by the same private key and having the same subject (happens in tests)
config.Entity.SerialNumber = serialNumber.String()

// Note: KeyUsageCRLSign is set only to generate empty CRLs for Desktop
// Access authentication with Windows.
keyUsage := x509.KeyUsageDigitalSignature | x509.KeyUsageCertSign | x509.KeyUsageCRLSign
if _, isRSA := config.Signer.Public().(*rsa.PublicKey); isRSA {
// The KeyEncipherment bit is necessary for RSA key exchanges
// https://datatracker.ietf.org/doc/html/rfc5280#section-4.2.1.3
keyUsage |= x509.KeyUsageKeyEncipherment
}

template := x509.Certificate{
SerialNumber: serialNumber,
Issuer: config.Entity,
Subject: config.Entity,
NotBefore: notBefore,
NotAfter: notAfter,
// Note: KeyUsageCRLSign is set only to generate empty CRLs for Desktop
// Access authentication with Windows.
KeyUsage: x509.KeyUsageKeyEncipherment | x509.KeyUsageDigitalSignature | x509.KeyUsageCertSign | x509.KeyUsageCRLSign,
SerialNumber: serialNumber,
Issuer: config.Entity,
Subject: config.Entity,
NotBefore: notBefore,
NotAfter: notAfter,
KeyUsage: keyUsage,
BasicConstraintsValid: true,
IsCA: true,
DNSNames: config.DNSNames,
Expand Down
2 changes: 1 addition & 1 deletion lib/utils/cert/selfsigned.go
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,7 @@ func GenerateSelfSignedCert(hostNames []string, ipAddresses []string, eku ...x50
Subject: entity,
NotBefore: notBefore,
NotAfter: notAfter,
KeyUsage: x509.KeyUsageKeyEncipherment | x509.KeyUsageDigitalSignature | x509.KeyUsageCertSign,
KeyUsage: x509.KeyUsageDigitalSignature | x509.KeyUsageCertSign,
ExtKeyUsage: eku,
BasicConstraintsValid: true,
IsCA: true,
Expand Down

0 comments on commit 20e6a71

Please sign in to comment.