2019-03-14 09:30:04 +01:00
|
|
|
package tls
|
|
|
|
|
|
|
|
import (
|
2019-09-13 19:28:04 +02:00
|
|
|
"context"
|
2019-03-14 09:30:04 +01:00
|
|
|
"crypto/tls"
|
2019-07-12 17:50:04 +02:00
|
|
|
"crypto/x509"
|
|
|
|
"encoding/pem"
|
2019-03-14 09:30:04 +01:00
|
|
|
"testing"
|
2019-06-21 17:18:05 +02:00
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
2019-07-12 17:50:04 +02:00
|
|
|
"github.com/stretchr/testify/require"
|
2019-03-14 09:30:04 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// LocalhostCert is a PEM-encoded TLS cert with SAN IPs
|
|
|
|
// "127.0.0.1" and "[::1]", expiring at Jan 29 16:00:00 2084 GMT.
|
|
|
|
// generated from src/crypto/tls:
|
|
|
|
// go run generate_cert.go --rsa-bits 1024 --host 127.0.0.1,::1,example.com --ca --start-date "Jan 1 00:00:00 1970" --duration=1000000h
|
|
|
|
var (
|
|
|
|
localhostCert = FileOrContent(`-----BEGIN CERTIFICATE-----
|
|
|
|
MIICEzCCAXygAwIBAgIQMIMChMLGrR+QvmQvpwAU6zANBgkqhkiG9w0BAQsFADAS
|
|
|
|
MRAwDgYDVQQKEwdBY21lIENvMCAXDTcwMDEwMTAwMDAwMFoYDzIwODQwMTI5MTYw
|
|
|
|
MDAwWjASMRAwDgYDVQQKEwdBY21lIENvMIGfMA0GCSqGSIb3DQEBAQUAA4GNADCB
|
|
|
|
iQKBgQDuLnQAI3mDgey3VBzWnB2L39JUU4txjeVE6myuDqkM/uGlfjb9SjY1bIw4
|
|
|
|
iA5sBBZzHi3z0h1YV8QPuxEbi4nW91IJm2gsvvZhIrCHS3l6afab4pZBl2+XsDul
|
|
|
|
rKBxKKtD1rGxlG4LjncdabFn9gvLZad2bSysqz/qTAUStTvqJQIDAQABo2gwZjAO
|
|
|
|
BgNVHQ8BAf8EBAMCAqQwEwYDVR0lBAwwCgYIKwYBBQUHAwEwDwYDVR0TAQH/BAUw
|
|
|
|
AwEB/zAuBgNVHREEJzAlggtleGFtcGxlLmNvbYcEfwAAAYcQAAAAAAAAAAAAAAAA
|
|
|
|
AAAAATANBgkqhkiG9w0BAQsFAAOBgQCEcetwO59EWk7WiJsG4x8SY+UIAA+flUI9
|
|
|
|
tyC4lNhbcF2Idq9greZwbYCqTTTr2XiRNSMLCOjKyI7ukPoPjo16ocHj+P3vZGfs
|
|
|
|
h1fIw3cSS2OolhloGw/XM6RWPWtPAlGykKLciQrBru5NAPvCMsb/I1DAceTiotQM
|
|
|
|
fblo6RBxUQ==
|
|
|
|
-----END CERTIFICATE-----`)
|
|
|
|
|
|
|
|
// LocalhostKey is the private key for localhostCert.
|
|
|
|
localhostKey = FileOrContent(`-----BEGIN RSA PRIVATE KEY-----
|
|
|
|
MIICXgIBAAKBgQDuLnQAI3mDgey3VBzWnB2L39JUU4txjeVE6myuDqkM/uGlfjb9
|
|
|
|
SjY1bIw4iA5sBBZzHi3z0h1YV8QPuxEbi4nW91IJm2gsvvZhIrCHS3l6afab4pZB
|
|
|
|
l2+XsDulrKBxKKtD1rGxlG4LjncdabFn9gvLZad2bSysqz/qTAUStTvqJQIDAQAB
|
|
|
|
AoGAGRzwwir7XvBOAy5tM/uV6e+Zf6anZzus1s1Y1ClbjbE6HXbnWWF/wbZGOpet
|
|
|
|
3Zm4vD6MXc7jpTLryzTQIvVdfQbRc6+MUVeLKwZatTXtdZrhu+Jk7hx0nTPy8Jcb
|
|
|
|
uJqFk541aEw+mMogY/xEcfbWd6IOkp+4xqjlFLBEDytgbIECQQDvH/E6nk+hgN4H
|
|
|
|
qzzVtxxr397vWrjrIgPbJpQvBsafG7b0dA4AFjwVbFLmQcj2PprIMmPcQrooz8vp
|
|
|
|
jy4SHEg1AkEA/v13/5M47K9vCxmb8QeD/asydfsgS5TeuNi8DoUBEmiSJwma7FXY
|
|
|
|
fFUtxuvL7XvjwjN5B30pNEbc6Iuyt7y4MQJBAIt21su4b3sjXNueLKH85Q+phy2U
|
|
|
|
fQtuUE9txblTu14q3N7gHRZB4ZMhFYyDy8CKrN2cPg/Fvyt0Xlp/DoCzjA0CQQDU
|
|
|
|
y2ptGsuSmgUtWj3NM9xuwYPm+Z/F84K6+ARYiZ6PYj013sovGKUFfYAqVXVlxtIX
|
|
|
|
qyUBnu3X9ps8ZfjLZO7BAkEAlT4R5Yl6cGhaJQYZHOde3JEMhNRcVFMO8dJDaFeo
|
|
|
|
f9Oeos0UUothgiDktdQHxdNEwLjQf7lJJBzV+5OtwswCWA==
|
|
|
|
-----END RSA PRIVATE KEY-----`)
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestTLSInStore(t *testing.T) {
|
2019-06-27 23:58:03 +02:00
|
|
|
dynamicConfigs := []*CertAndStores{{
|
|
|
|
Certificate: Certificate{
|
|
|
|
CertFile: localhostCert,
|
|
|
|
KeyFile: localhostKey,
|
|
|
|
},
|
|
|
|
}}
|
2019-03-14 09:30:04 +01:00
|
|
|
|
|
|
|
tlsManager := NewManager()
|
2019-09-13 19:28:04 +02:00
|
|
|
tlsManager.UpdateConfigs(context.Background(), nil, nil, dynamicConfigs)
|
2019-03-14 09:30:04 +01:00
|
|
|
|
|
|
|
certs := tlsManager.GetStore("default").DynamicCerts.Get().(map[string]*tls.Certificate)
|
|
|
|
if len(certs) == 0 {
|
|
|
|
t.Fatal("got error: default store must have TLS certificates.")
|
|
|
|
}
|
|
|
|
}
|
2019-06-21 16:32:04 +02:00
|
|
|
|
|
|
|
func TestTLSInvalidStore(t *testing.T) {
|
2019-06-27 23:58:03 +02:00
|
|
|
dynamicConfigs := []*CertAndStores{{
|
|
|
|
Certificate: Certificate{
|
|
|
|
CertFile: localhostCert,
|
|
|
|
KeyFile: localhostKey,
|
|
|
|
},
|
|
|
|
}}
|
2019-06-21 16:32:04 +02:00
|
|
|
|
|
|
|
tlsManager := NewManager()
|
2019-09-13 19:28:04 +02:00
|
|
|
tlsManager.UpdateConfigs(context.Background(),
|
|
|
|
map[string]Store{
|
|
|
|
"default": {
|
|
|
|
DefaultCertificate: &Certificate{
|
|
|
|
CertFile: "/wrong",
|
|
|
|
KeyFile: "/wrong",
|
|
|
|
},
|
2019-06-21 16:32:04 +02:00
|
|
|
},
|
2019-09-13 19:28:04 +02:00
|
|
|
}, nil, dynamicConfigs)
|
2019-06-21 16:32:04 +02:00
|
|
|
|
|
|
|
certs := tlsManager.GetStore("default").DynamicCerts.Get().(map[string]*tls.Certificate)
|
|
|
|
if len(certs) == 0 {
|
|
|
|
t.Fatal("got error: default store must have TLS certificates.")
|
|
|
|
}
|
|
|
|
}
|
2019-06-21 17:18:05 +02:00
|
|
|
|
|
|
|
func TestManager_Get(t *testing.T) {
|
2019-06-27 23:58:03 +02:00
|
|
|
dynamicConfigs := []*CertAndStores{{
|
|
|
|
Certificate: Certificate{
|
|
|
|
CertFile: localhostCert,
|
|
|
|
KeyFile: localhostKey,
|
|
|
|
},
|
|
|
|
}}
|
|
|
|
|
|
|
|
tlsConfigs := map[string]Options{
|
2019-06-21 17:18:05 +02:00
|
|
|
"foo": {MinVersion: "VersionTLS12"},
|
|
|
|
"bar": {MinVersion: "VersionTLS11"},
|
|
|
|
}
|
|
|
|
|
|
|
|
testCases := []struct {
|
|
|
|
desc string
|
|
|
|
tlsOptionsName string
|
|
|
|
expectedMinVersion uint16
|
|
|
|
expectedError bool
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
desc: "Get a tls config from a valid name",
|
|
|
|
tlsOptionsName: "foo",
|
|
|
|
expectedMinVersion: uint16(tls.VersionTLS12),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "Get another tls config from a valid name",
|
|
|
|
tlsOptionsName: "bar",
|
|
|
|
expectedMinVersion: uint16(tls.VersionTLS11),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "Get an tls config from an invalid name",
|
|
|
|
tlsOptionsName: "unknown",
|
|
|
|
expectedError: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "Get an tls config from unexisting 'default' name",
|
|
|
|
tlsOptionsName: "default",
|
|
|
|
expectedError: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
tlsManager := NewManager()
|
2019-09-13 19:28:04 +02:00
|
|
|
tlsManager.UpdateConfigs(context.Background(), nil, tlsConfigs, dynamicConfigs)
|
2019-06-21 17:18:05 +02:00
|
|
|
|
|
|
|
for _, test := range testCases {
|
|
|
|
test := test
|
|
|
|
t.Run(test.desc, func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
config, err := tlsManager.Get("default", test.tlsOptionsName)
|
|
|
|
if test.expectedError {
|
|
|
|
assert.Error(t, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, config.MinVersion, test.expectedMinVersion)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2019-07-12 17:50:04 +02:00
|
|
|
|
|
|
|
func TestClientAuth(t *testing.T) {
|
|
|
|
tlsConfigs := map[string]Options{
|
2019-07-19 11:52:04 +02:00
|
|
|
"eca": {
|
|
|
|
ClientAuth: ClientAuth{},
|
|
|
|
},
|
|
|
|
"ecat": {
|
|
|
|
ClientAuth: ClientAuth{ClientAuthType: ""},
|
|
|
|
},
|
|
|
|
"ncc": {
|
|
|
|
ClientAuth: ClientAuth{ClientAuthType: "NoClientCert"},
|
|
|
|
},
|
|
|
|
"rcc": {
|
|
|
|
ClientAuth: ClientAuth{ClientAuthType: "RequestClientCert"},
|
|
|
|
},
|
|
|
|
"racc": {
|
|
|
|
ClientAuth: ClientAuth{ClientAuthType: "RequireAnyClientCert"},
|
|
|
|
},
|
2019-07-12 17:50:04 +02:00
|
|
|
"vccig": {
|
|
|
|
ClientAuth: ClientAuth{
|
|
|
|
CAFiles: []FileOrContent{localhostCert},
|
|
|
|
ClientAuthType: "VerifyClientCertIfGiven",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
"vccigwca": {
|
|
|
|
ClientAuth: ClientAuth{ClientAuthType: "VerifyClientCertIfGiven"},
|
|
|
|
},
|
2019-07-19 11:52:04 +02:00
|
|
|
"ravcc": {
|
|
|
|
ClientAuth: ClientAuth{ClientAuthType: "RequireAndVerifyClientCert"},
|
|
|
|
},
|
2019-07-12 17:50:04 +02:00
|
|
|
"ravccwca": {
|
|
|
|
ClientAuth: ClientAuth{
|
|
|
|
CAFiles: []FileOrContent{localhostCert},
|
|
|
|
ClientAuthType: "RequireAndVerifyClientCert",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
"ravccwbca": {
|
|
|
|
ClientAuth: ClientAuth{
|
|
|
|
CAFiles: []FileOrContent{"Bad content"},
|
|
|
|
ClientAuthType: "RequireAndVerifyClientCert",
|
|
|
|
},
|
|
|
|
},
|
2019-07-19 11:52:04 +02:00
|
|
|
"ucat": {
|
|
|
|
ClientAuth: ClientAuth{ClientAuthType: "Unknown"},
|
|
|
|
},
|
2019-07-12 17:50:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
block, _ := pem.Decode([]byte(localhostCert))
|
|
|
|
cert, err := x509.ParseCertificate(block.Bytes)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
testCases := []struct {
|
|
|
|
desc string
|
|
|
|
tlsOptionsName string
|
|
|
|
expectedClientAuth tls.ClientAuthType
|
|
|
|
expectedRawSubject []byte
|
2019-07-19 11:52:04 +02:00
|
|
|
expectedError bool
|
2019-07-12 17:50:04 +02:00
|
|
|
}{
|
|
|
|
{
|
|
|
|
desc: "Empty ClientAuth option should get a tls.NoClientCert (default value)",
|
|
|
|
tlsOptionsName: "eca",
|
|
|
|
expectedClientAuth: tls.NoClientCert,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "Empty ClientAuthType option should get a tls.NoClientCert (default value)",
|
|
|
|
tlsOptionsName: "ecat",
|
|
|
|
expectedClientAuth: tls.NoClientCert,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "NoClientCert option should get a tls.NoClientCert as ClientAuthType",
|
|
|
|
tlsOptionsName: "ncc",
|
|
|
|
expectedClientAuth: tls.NoClientCert,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "RequestClientCert option should get a tls.RequestClientCert as ClientAuthType",
|
|
|
|
tlsOptionsName: "rcc",
|
|
|
|
expectedClientAuth: tls.RequestClientCert,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "RequireAnyClientCert option should get a tls.RequireAnyClientCert as ClientAuthType",
|
|
|
|
tlsOptionsName: "racc",
|
|
|
|
expectedClientAuth: tls.RequireAnyClientCert,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "VerifyClientCertIfGiven option should get a tls.VerifyClientCertIfGiven as ClientAuthType",
|
|
|
|
tlsOptionsName: "vccig",
|
|
|
|
expectedClientAuth: tls.VerifyClientCertIfGiven,
|
|
|
|
},
|
|
|
|
{
|
2019-07-19 11:52:04 +02:00
|
|
|
desc: "VerifyClientCertIfGiven option without CAFiles yields a default ClientAuthType (NoClientCert)",
|
2019-07-12 17:50:04 +02:00
|
|
|
tlsOptionsName: "vccigwca",
|
|
|
|
expectedClientAuth: tls.NoClientCert,
|
2019-07-19 11:52:04 +02:00
|
|
|
expectedError: true,
|
2019-07-12 17:50:04 +02:00
|
|
|
},
|
|
|
|
{
|
2019-07-19 11:52:04 +02:00
|
|
|
desc: "RequireAndVerifyClientCert option without CAFiles yields a default ClientAuthType (NoClientCert)",
|
2019-07-12 17:50:04 +02:00
|
|
|
tlsOptionsName: "ravcc",
|
|
|
|
expectedClientAuth: tls.NoClientCert,
|
2019-07-19 11:52:04 +02:00
|
|
|
expectedError: true,
|
2019-07-12 17:50:04 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "RequireAndVerifyClientCert option should get a tls.RequireAndVerifyClientCert as ClientAuthType with CA files",
|
|
|
|
tlsOptionsName: "ravccwca",
|
|
|
|
expectedClientAuth: tls.RequireAndVerifyClientCert,
|
|
|
|
expectedRawSubject: cert.RawSubject,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "Unknown option yields a default ClientAuthType (NoClientCert)",
|
|
|
|
tlsOptionsName: "ucat",
|
|
|
|
expectedClientAuth: tls.NoClientCert,
|
2019-07-19 11:52:04 +02:00
|
|
|
expectedError: true,
|
2019-07-12 17:50:04 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "Bad CA certificate content yields a default ClientAuthType (NoClientCert)",
|
|
|
|
tlsOptionsName: "ravccwbca",
|
|
|
|
expectedClientAuth: tls.NoClientCert,
|
2019-07-19 11:52:04 +02:00
|
|
|
expectedError: true,
|
2019-07-12 17:50:04 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
tlsManager := NewManager()
|
2019-09-13 19:28:04 +02:00
|
|
|
tlsManager.UpdateConfigs(context.Background(), nil, tlsConfigs, nil)
|
2019-07-12 17:50:04 +02:00
|
|
|
|
|
|
|
for _, test := range testCases {
|
|
|
|
test := test
|
|
|
|
t.Run(test.desc, func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
config, err := tlsManager.Get("default", test.tlsOptionsName)
|
2019-07-19 11:52:04 +02:00
|
|
|
|
|
|
|
if test.expectedError {
|
|
|
|
assert.Error(t, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-07-12 17:50:04 +02:00
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
if test.expectedRawSubject != nil {
|
|
|
|
subjects := config.ClientCAs.Subjects()
|
|
|
|
assert.Len(t, subjects, 1)
|
|
|
|
assert.Equal(t, subjects[0], test.expectedRawSubject)
|
|
|
|
}
|
|
|
|
|
|
|
|
assert.Equal(t, config.ClientAuth, test.expectedClientAuth)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|