Merge pull request #101707 from enj/enj/i/bad_cadata
client-go transport: assert that final CA data is valid
This commit is contained in:
@@ -17,7 +17,6 @@ vendor/k8s.io/apiserver/pkg/storage
|
|||||||
vendor/k8s.io/apiserver/pkg/storage/cacher
|
vendor/k8s.io/apiserver/pkg/storage/cacher
|
||||||
vendor/k8s.io/apiserver/pkg/storage/tests
|
vendor/k8s.io/apiserver/pkg/storage/tests
|
||||||
vendor/k8s.io/apiserver/pkg/storage/value/encrypt/envelope
|
vendor/k8s.io/apiserver/pkg/storage/value/encrypt/envelope
|
||||||
vendor/k8s.io/apiserver/pkg/util/webhook
|
|
||||||
vendor/k8s.io/apiserver/pkg/util/wsstream
|
vendor/k8s.io/apiserver/pkg/util/wsstream
|
||||||
vendor/k8s.io/client-go/rest
|
vendor/k8s.io/client-go/rest
|
||||||
vendor/k8s.io/client-go/rest/watch
|
vendor/k8s.io/client-go/rest/watch
|
||||||
|
@@ -67,34 +67,6 @@ ID3CIiCF8hhF5C2hjrW0LTJ6zTlg1c1K0NmmUL1ucsfzEk//c7GsU8dk+FYmtW9A
|
|||||||
VzryJj1NmnSqA3zd3jBMuK37Ei3pRvVbO7Uxf14=
|
VzryJj1NmnSqA3zd3jBMuK37Ei3pRvVbO7Uxf14=
|
||||||
-----END CERTIFICATE-----`)
|
-----END CERTIFICATE-----`)
|
||||||
|
|
||||||
var badCAKey = []byte(`-----BEGIN RSA PRIVATE KEY-----
|
|
||||||
MIIEowIBAAKCAQEAsmGBQnmBK88DdHezRXed0pX0vtbxMP6WEwaxW1d6x7jOafn2
|
|
||||||
HbfxuxC/B1dpewpKswBlQTi4/0LqjUi2/YJIyVJZXGdFRaR5zqYVK4LYNHSBvY8y
|
|
||||||
9wyupUOKGiP6fx9bMAXOzafdK0LR7C8P5q9+zxRcjU8G5sK9mnd+oNLKMZJ1gaMU
|
|
||||||
3KMYi3gnxi5ebNxxP0LAaPqY+yfpjSm2OZkS/rcMWMXsq42ERPvbclf0rY5o+uau
|
|
||||||
XXaNilkjJhUUoEk0bOlqflsh2ol+WVwhS3XIO9MORwspP58drn574JcwhMFlM7aU
|
|
||||||
kT4/hVoqnSaOmSdLR3Op3OmQSq+n+J9cbJiVfQIDAQABAoIBAQCZd8n9pwu65R/T
|
|
||||||
1CgoXAEsbFdk2QgpXt8+/0MXkuvPaPAtvSBB8T/H8WBosIvPj8s0teJneqWu96NU
|
|
||||||
ansFIFH+4xp+pVqz0A37/Ge6R5g7iQEWVV1Dr2WSSclHNC0PsaqCZnzF8uYVkieJ
|
|
||||||
S/QiRFqVTq9R4+vMHT+C5cvMEY5jlm5LA35UKOZ0SQRHJ717WJZK+p+XszljaI9g
|
|
||||||
9sCZX0uR25nECalhwZVbaK6IB4u1nBM6+67j2sOUV/7udyPbDlAfRE15dQxBR+rW
|
|
||||||
Uu30jQz6+GMsPh1bHlfDHy4L/FFPCff/ZI3fbbfuvfUMJHH3G/eJvi+QPtlQN6Xp
|
|
||||||
H9xl/NgBAoGBANdU96dAl4N0LlkEagzGXXSQMgxDbuMkOHlDGH1Vm0P+Nq7tZKwF
|
|
||||||
EQKvVa6QPaSiAzgeuUwPkiVlIJZ6lMzzutRMKu3fSYnM57X45dnKWfs9AsskDSMp
|
|
||||||
Pj2LQ1RBo8Pgd8H/aFDVmCjBdn6w/6xrAPn4TvpYM6mJyQ+KbcMINrXZAoGBANQS
|
|
||||||
ADwajz00IXNxE6A86mvQZ68fphNNoMByI6q5bA+T8AorlQ9YaofYfb1ezmHIh6nt
|
|
||||||
QPQIHATYF6BmY23ypgHF/aE+lcgK7Nej4u+3Lc9ImJ/0o5wzvxUpX9XFsjOjAGEz
|
|
||||||
TentETTbLnp6a3f+bnXAOaM/ofQNPvnqiQnR0OJFAoGAC7Jp4YP4twNQoTVELX15
|
|
||||||
BiPvFAt1spD9IFkss2I7FO5yOf5bQZzk16h+lwTu1EqYsiu5FRCjd7SOmJ4AB0IW
|
|
||||||
HAInMtS2Qe4HiDMFCVecm7EsvawvqoFLCDzQY3tNUg6XcspU+E8h/NTFgwxKVytY
|
|
||||||
2jtKzv6Lj+IUMevrGnUPw8ECgYAGnuE+/x1FreD1d6xDLmOrJgB2qShIJf5Ew8t1
|
|
||||||
QwCqo9W0m5O1vO7mes3CIbmTt+z0UyHZ/H7Tb+Oc8FVeU1r3ZzT52bhXXG/0c3tc
|
|
||||||
PH3DoOKS69JHyB3JDVeeluNvVUFnx3BBQ1NsMQOMc1Hzlw/fwTaLcCsgMWGr77SD
|
|
||||||
h/dbeQKBgD9wanYnE7uQAP9N49FDtCiiMwFyYc8QwULU2jD6umfByjmvo4ocqDE/
|
|
||||||
sMwQmu62qrQlhHPMXh28zsobyC3DpkgutlTB9U1xyMoElFGeEBbpd3JhTZFKs7Hd
|
|
||||||
RggKDB1828evoGHyfA+Tu1+uh0viEuP0yfIMjRABbJaLjTYAx1YG
|
|
||||||
-----END RSA PRIVATE KEY-----`)
|
|
||||||
|
|
||||||
var badCACert = []byte(`-----BEGIN CERTIFICATE-----
|
var badCACert = []byte(`-----BEGIN CERTIFICATE-----
|
||||||
MIIDGTCCAgGgAwIBAgIUINBaI0NGgSo4UqKHuPd/HSBO38EwDQYJKoZIhvcNAQEL
|
MIIDGTCCAgGgAwIBAgIUINBaI0NGgSo4UqKHuPd/HSBO38EwDQYJKoZIhvcNAQEL
|
||||||
BQAwGzEZMBcGA1UEAwwQd2ViaG9va190ZXN0c19jYTAgFw0yMDEwMDcxMjMxNDFa
|
BQAwGzEZMBcGA1UEAwwQd2ViaG9va190ZXN0c19jYTAgFw0yMDEwMDcxMjMxNDFa
|
||||||
|
@@ -163,15 +163,41 @@ func TestKubeConfigFile(t *testing.T) {
|
|||||||
errRegex: fmt.Sprintf(errMissingCertPath, "certificate-authority", badCAPath, "", badCAPath),
|
errRegex: fmt.Sprintf(errMissingCertPath, "certificate-authority", badCAPath, "", badCAPath),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
test: "cluster with invalid CA certificate ",
|
test: "cluster with invalid CA certificate",
|
||||||
cluster: &v1.NamedCluster{
|
cluster: &v1.NamedCluster{
|
||||||
Cluster: v1.Cluster{
|
Cluster: v1.Cluster{
|
||||||
Server: namedCluster.Cluster.Server,
|
Server: namedCluster.Cluster.Server,
|
||||||
CertificateAuthorityData: caKey,
|
CertificateAuthorityData: caKey, // pretend user put caKey here instead of caCert
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
user: &defaultUser,
|
user: &defaultUser,
|
||||||
errRegex: "", // Not an error at parse time, only when using the webhook
|
errRegex: "unable to load root certificates: no valid certificate authority data seen",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
test: "cluster with invalid CA certificate - no PEM",
|
||||||
|
cluster: &v1.NamedCluster{
|
||||||
|
Cluster: v1.Cluster{
|
||||||
|
Server: namedCluster.Cluster.Server,
|
||||||
|
CertificateAuthorityData: []byte(`not a cert`),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
user: &defaultUser,
|
||||||
|
errRegex: "unable to load root certificates: unable to parse bytes as PEM block",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
test: "cluster with invalid CA certificate - parse error",
|
||||||
|
cluster: &v1.NamedCluster{
|
||||||
|
Cluster: v1.Cluster{
|
||||||
|
Server: namedCluster.Cluster.Server,
|
||||||
|
CertificateAuthorityData: []byte(`
|
||||||
|
-----BEGIN CERTIFICATE-----
|
||||||
|
MIIDGTCCAgGgAwIBAgIUOS2M
|
||||||
|
-----END CERTIFICATE-----
|
||||||
|
`),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
user: &defaultUser,
|
||||||
|
errRegex: "unable to load root certificates: failed to parse certificate: asn1: syntax error: data truncated",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
test: "user with invalid client certificate path",
|
test: "user with invalid client certificate path",
|
||||||
@@ -242,6 +268,7 @@ func TestKubeConfigFile(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for _, tt := range tests {
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.test, func(t *testing.T) {
|
||||||
// Use a closure so defer statements trigger between loop iterations.
|
// Use a closure so defer statements trigger between loop iterations.
|
||||||
err := func() error {
|
err := func() error {
|
||||||
kubeConfig := v1.Config{}
|
kubeConfig := v1.Config{}
|
||||||
@@ -282,6 +309,7 @@ func TestKubeConfigFile(t *testing.T) {
|
|||||||
t.Errorf("%s: unexpected error message to match:\n Expected: %s\n Actual: %s", tt.test, tt.errRegex, err.Error())
|
t.Errorf("%s: unexpected error message to match:\n Expected: %s\n Actual: %s", tt.test, tt.errRegex, err.Error())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -20,6 +20,7 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"crypto/x509"
|
"crypto/x509"
|
||||||
|
"encoding/pem"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"net/http"
|
"net/http"
|
||||||
@@ -79,7 +80,11 @@ func TLSConfigFor(c *Config) (*tls.Config, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if c.HasCA() {
|
if c.HasCA() {
|
||||||
tlsConfig.RootCAs = rootCertPool(c.TLS.CAData)
|
rootCAs, err := rootCertPool(c.TLS.CAData)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("unable to load root certificates: %w", err)
|
||||||
|
}
|
||||||
|
tlsConfig.RootCAs = rootCAs
|
||||||
}
|
}
|
||||||
|
|
||||||
var staticCert *tls.Certificate
|
var staticCert *tls.Certificate
|
||||||
@@ -176,18 +181,41 @@ func dataFromSliceOrFile(data []byte, file string) ([]byte, error) {
|
|||||||
|
|
||||||
// rootCertPool returns nil if caData is empty. When passed along, this will mean "use system CAs".
|
// rootCertPool returns nil if caData is empty. When passed along, this will mean "use system CAs".
|
||||||
// When caData is not empty, it will be the ONLY information used in the CertPool.
|
// When caData is not empty, it will be the ONLY information used in the CertPool.
|
||||||
func rootCertPool(caData []byte) *x509.CertPool {
|
func rootCertPool(caData []byte) (*x509.CertPool, error) {
|
||||||
// What we really want is a copy of x509.systemRootsPool, but that isn't exposed. It's difficult to build (see the go
|
// What we really want is a copy of x509.systemRootsPool, but that isn't exposed. It's difficult to build (see the go
|
||||||
// code for a look at the platform specific insanity), so we'll use the fact that RootCAs == nil gives us the system values
|
// code for a look at the platform specific insanity), so we'll use the fact that RootCAs == nil gives us the system values
|
||||||
// It doesn't allow trusting either/or, but hopefully that won't be an issue
|
// It doesn't allow trusting either/or, but hopefully that won't be an issue
|
||||||
if len(caData) == 0 {
|
if len(caData) == 0 {
|
||||||
return nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// if we have caData, use it
|
// if we have caData, use it
|
||||||
certPool := x509.NewCertPool()
|
certPool := x509.NewCertPool()
|
||||||
certPool.AppendCertsFromPEM(caData)
|
if ok := certPool.AppendCertsFromPEM(caData); !ok {
|
||||||
return certPool
|
return nil, createErrorParsingCAData(caData)
|
||||||
|
}
|
||||||
|
return certPool, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// createErrorParsingCAData ALWAYS returns an error. We call it because know we failed to AppendCertsFromPEM
|
||||||
|
// but we don't know the specific error because that API is just true/false
|
||||||
|
func createErrorParsingCAData(pemCerts []byte) error {
|
||||||
|
for len(pemCerts) > 0 {
|
||||||
|
var block *pem.Block
|
||||||
|
block, pemCerts = pem.Decode(pemCerts)
|
||||||
|
if block == nil {
|
||||||
|
return fmt.Errorf("unable to parse bytes as PEM block")
|
||||||
|
}
|
||||||
|
|
||||||
|
if block.Type != "CERTIFICATE" || len(block.Headers) != 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := x509.ParseCertificate(block.Bytes); err != nil {
|
||||||
|
return fmt.Errorf("failed to parse certificate: %w", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return fmt.Errorf("no valid certificate authority data seen")
|
||||||
}
|
}
|
||||||
|
|
||||||
// WrapperFunc wraps an http.RoundTripper when a new transport
|
// WrapperFunc wraps an http.RoundTripper when a new transport
|
||||||
|
@@ -142,6 +142,14 @@ func TestNew(t *testing.T) {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
"bad ca data transport": {
|
||||||
|
Err: true,
|
||||||
|
Config: &Config{
|
||||||
|
TLS: TLSConfig{
|
||||||
|
CAData: []byte(rootCACert + "this is not valid"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
"ca data overriding bad ca file transport": {
|
"ca data overriding bad ca file transport": {
|
||||||
TLS: true,
|
TLS: true,
|
||||||
Config: &Config{
|
Config: &Config{
|
||||||
|
@@ -39,26 +39,27 @@ func TestInsecurePodLogs(t *testing.T) {
|
|||||||
ModifyServerRunOptions: func(opts *options.ServerRunOptions) {
|
ModifyServerRunOptions: func(opts *options.ServerRunOptions) {
|
||||||
opts.GenericServerRunOptions.MaxRequestBodyBytes = 1024 * 1024
|
opts.GenericServerRunOptions.MaxRequestBodyBytes = 1024 * 1024
|
||||||
// I have no idea what this cert is, but it doesn't matter, we just want something that always fails validation
|
// I have no idea what this cert is, but it doesn't matter, we just want something that always fails validation
|
||||||
opts.KubeletConfig.CAData = []byte(` -----BEGIN CERTIFICATE-----
|
opts.KubeletConfig.CAData = []byte(`
|
||||||
MIIDMDCCAhigAwIBAgIIHNPD7sig7YIwDQYJKoZIhvcNAQELBQAwNjESMBAGA1UE
|
-----BEGIN CERTIFICATE-----
|
||||||
CxMJb3BlbnNoaWZ0MSAwHgYDVQQDExdhZG1pbi1rdWJlY29uZmlnLXNpZ25lcjAe
|
MIIDMDCCAhigAwIBAgIIHNPD7sig7YIwDQYJKoZIhvcNAQELBQAwNjESMBAGA1UE
|
||||||
Fw0xOTA1MzAxNTA3MzlaFw0yOTA1MjcxNTA3MzlaMDYxEjAQBgNVBAsTCW9wZW5z
|
CxMJb3BlbnNoaWZ0MSAwHgYDVQQDExdhZG1pbi1rdWJlY29uZmlnLXNpZ25lcjAe
|
||||||
aGlmdDEgMB4GA1UEAxMXYWRtaW4ta3ViZWNvbmZpZy1zaWduZXIwggEiMA0GCSqG
|
Fw0xOTA1MzAxNTA3MzlaFw0yOTA1MjcxNTA3MzlaMDYxEjAQBgNVBAsTCW9wZW5z
|
||||||
SIb3DQEBAQUAA4IBDwAwggEKAoIBAQD0dHk23lHRcuq06FzYDOl9J9+s8pnGxqA3
|
aGlmdDEgMB4GA1UEAxMXYWRtaW4ta3ViZWNvbmZpZy1zaWduZXIwggEiMA0GCSqG
|
||||||
IPcARI6ag/98aYe3ENwAB5e1i7AU2F2WiDZgj444w374XLdVgIK8zgQEm9yoqrlc
|
SIb3DQEBAQUAA4IBDwAwggEKAoIBAQD0dHk23lHRcuq06FzYDOl9J9+s8pnGxqA3
|
||||||
+/ayO7ceKklrKHOMwh63LvGLEOqzhol2nFmBhXAZt+HyIoZHXN0IqlA92196+Dml
|
IPcARI6ag/98aYe3ENwAB5e1i7AU2F2WiDZgj444w374XLdVgIK8zgQEm9yoqrlc
|
||||||
0WOn1F4ce6JbAtEceFHPgLeI7KFmVaPz2796pBXh23ii6r7WvV1Rn9MKlMSBJQR4
|
+/ayO7ceKklrKHOMwh63LvGLEOqzhol2nFmBhXAZt+HyIoZHXN0IqlA92196+Dml
|
||||||
0LZzu9/j+GdnFXewdLAAMfgPzwEqv6h3PzvtUCjgdraHEm8Rs7s15S3PUmLK4RQS
|
0WOn1F4ce6JbAtEceFHPgLeI7KFmVaPz2796pBXh23ii6r7WvV1Rn9MKlMSBJQR4
|
||||||
PsThx5BhJEGd/W6EzQ3BKoQfochhu3mnAQtW1J07CullySQ5Gg9fAgMBAAGjQjBA
|
0LZzu9/j+GdnFXewdLAAMfgPzwEqv6h3PzvtUCjgdraHEm8Rs7s15S3PUmLK4RQS
|
||||||
MA4GA1UdDwEB/wQEAwICpDAPBgNVHRMBAf8EBTADAQH/MB0GA1UdDgQWBBQkTaaw
|
PsThx5BhJEGd/W6EzQ3BKoQfochhu3mnAQtW1J07CullySQ5Gg9fAgMBAAGjQjBA
|
||||||
YJSZ5k2Wd+OsM4GFMTGdqzANBgkqhkiG9w0BAQsFAAOCAQEAHK7+zBZPLqK+f9DT
|
MA4GA1UdDwEB/wQEAwICpDAPBgNVHRMBAf8EBTADAQH/MB0GA1UdDgQWBBQkTaaw
|
||||||
UEnpwRmZ0aeGS4YgbGIkqpjxJymVOwkRd5A1wslvVfGZ6yOQthF6KlCmqnPyJJMR
|
YJSZ5k2Wd+OsM4GFMTGdqzANBgkqhkiG9w0BAQsFAAOCAQEAHK7+zBZPLqK+f9DT
|
||||||
I7FHw8j0h2ci90fEQ6IS90Y/ZJXkcgiK9Ncwa35GFGs8QrBxN4leGhtm84BnnBHN
|
UEnpwRmZ0aeGS4YgbGIkqpjxJymVOwkRd5A1wslvVfGZ6yOQthF6KlCmqnPyJJMR
|
||||||
cTWpa4zcBwru0CRG7iHc66VX16X8jHB1iFeZ5W/FgY4MsE+G1Vze4mCXSPVI4BZ2
|
I7FHw8j0h2ci90fEQ6IS90Y/ZJXkcgiK9Ncwa35GFGs8QrBxN4leGhtm84BnnBHN
|
||||||
/qlAgogjBivvSwQ9SFuCszg7IPjvT2ksm+Cf+8eT4YBqW41F85vBGR+FYK14yIla
|
cTWpa4zcBwru0CRG7iHc66VX16X8jHB1iFeZ5W/FgY4MsE+G1Vze4mCXSPVI4BZ2
|
||||||
Bgqc+dJN9xS9Ah5gLiGQJ6C4niUA11piCpvMsy+j/LQ1Erx47KMar5fuMXYk7iPq
|
/qlAgogjBivvSwQ9SFuCszg7IPjvT2ksm+Cf+8eT4YBqW41F85vBGR+FYK14yIla
|
||||||
1vqIwg==
|
Bgqc+dJN9xS9Ah5gLiGQJ6C4niUA11piCpvMsy+j/LQ1Erx47KMar5fuMXYk7iPq
|
||||||
-----END CERTIFICATE-----
|
1vqIwg==
|
||||||
|
-----END CERTIFICATE-----
|
||||||
`)
|
`)
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
Reference in New Issue
Block a user