2018-10-05 21:48:36 +00:00
|
|
|
package authority
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/smallstep/assert"
|
2018-10-12 06:03:00 +00:00
|
|
|
"github.com/smallstep/ca-component/provisioner"
|
2018-10-05 21:48:36 +00:00
|
|
|
"github.com/smallstep/cli/crypto/tlsutil"
|
|
|
|
"github.com/smallstep/cli/crypto/x509util"
|
|
|
|
stepJOSE "github.com/smallstep/cli/jose"
|
2018-10-09 06:25:18 +00:00
|
|
|
jose "gopkg.in/square/go-jose.v2"
|
2018-10-05 21:48:36 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestConfigValidate(t *testing.T) {
|
|
|
|
maxjwk, err := stepJOSE.ParseKey("testdata/secrets/max_pub.jwk")
|
|
|
|
assert.FatalError(t, err)
|
|
|
|
clijwk, err := stepJOSE.ParseKey("testdata/secrets/step_cli_key_pub.jwk")
|
|
|
|
assert.FatalError(t, err)
|
|
|
|
ac := &AuthConfig{
|
2018-10-12 06:03:00 +00:00
|
|
|
Provisioners: []*provisioner.Provisioner{
|
2018-10-05 21:48:36 +00:00
|
|
|
{
|
|
|
|
Issuer: "Max",
|
|
|
|
Type: "JWK",
|
|
|
|
Key: maxjwk,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Issuer: "step-cli",
|
|
|
|
Type: "JWK",
|
|
|
|
Key: clijwk,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
type ConfigValidateTest struct {
|
|
|
|
config *Config
|
|
|
|
err error
|
|
|
|
tls tlsutil.TLSOptions
|
|
|
|
}
|
|
|
|
tests := map[string]func(*testing.T) ConfigValidateTest{
|
|
|
|
"empty-address": func(t *testing.T) ConfigValidateTest {
|
|
|
|
return ConfigValidateTest{
|
|
|
|
config: &Config{
|
|
|
|
Root: "testdata/secrets/root_ca.crt",
|
|
|
|
IntermediateCert: "testdata/secrets/intermediate_ca.crt",
|
|
|
|
IntermediateKey: "testdata/secrets/intermediate_ca_key",
|
|
|
|
DNSNames: []string{"test.smallstep.com"},
|
|
|
|
Password: "pass",
|
|
|
|
AuthorityConfig: ac,
|
|
|
|
},
|
|
|
|
err: errors.New("address cannot be empty"),
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"empty-root": func(t *testing.T) ConfigValidateTest {
|
|
|
|
return ConfigValidateTest{
|
|
|
|
config: &Config{
|
|
|
|
Address: "127.0.0.1",
|
|
|
|
IntermediateCert: "testdata/secrets/intermediate_ca.crt",
|
|
|
|
IntermediateKey: "testdata/secrets/intermediate_ca_key",
|
|
|
|
DNSNames: []string{"test.smallstep.com"},
|
|
|
|
Password: "pass",
|
|
|
|
AuthorityConfig: ac,
|
|
|
|
},
|
|
|
|
err: errors.New("root cannot be empty"),
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"empty-intermediate-cert": func(t *testing.T) ConfigValidateTest {
|
|
|
|
return ConfigValidateTest{
|
|
|
|
config: &Config{
|
|
|
|
Address: "127.0.0.1",
|
|
|
|
Root: "testdata/secrets/root_ca.crt",
|
|
|
|
IntermediateKey: "testdata/secrets/intermediate_ca_key",
|
|
|
|
DNSNames: []string{"test.smallstep.com"},
|
|
|
|
Password: "pass",
|
|
|
|
AuthorityConfig: ac,
|
|
|
|
},
|
|
|
|
err: errors.New("crt cannot be empty"),
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"empty-intermediate-key": func(t *testing.T) ConfigValidateTest {
|
|
|
|
return ConfigValidateTest{
|
|
|
|
config: &Config{
|
|
|
|
Address: "127.0.0.1",
|
|
|
|
Root: "testdata/secrets/root_ca.crt",
|
|
|
|
IntermediateCert: "testdata/secrets/intermediate_ca.crt",
|
|
|
|
DNSNames: []string{"test.smallstep.com"},
|
|
|
|
Password: "pass",
|
|
|
|
AuthorityConfig: ac,
|
|
|
|
},
|
|
|
|
err: errors.New("key cannot be empty"),
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"empty-dnsNames": func(t *testing.T) ConfigValidateTest {
|
|
|
|
return ConfigValidateTest{
|
|
|
|
config: &Config{
|
|
|
|
Address: "127.0.0.1",
|
|
|
|
Root: "testdata/secrets/root_ca.crt",
|
|
|
|
IntermediateCert: "testdata/secrets/intermediate_ca.crt",
|
|
|
|
IntermediateKey: "testdata/secrets/intermediate_ca_key",
|
|
|
|
Password: "pass",
|
|
|
|
AuthorityConfig: ac,
|
|
|
|
},
|
|
|
|
err: errors.New("dnsNames cannot be empty"),
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"empty-TLS": func(t *testing.T) ConfigValidateTest {
|
|
|
|
return ConfigValidateTest{
|
|
|
|
config: &Config{
|
|
|
|
Address: "127.0.0.1",
|
|
|
|
Root: "testdata/secrets/root_ca.crt",
|
|
|
|
IntermediateCert: "testdata/secrets/intermediate_ca.crt",
|
|
|
|
IntermediateKey: "testdata/secrets/intermediate_ca_key",
|
|
|
|
DNSNames: []string{"test.smallstep.com"},
|
|
|
|
Password: "pass",
|
|
|
|
AuthorityConfig: ac,
|
|
|
|
},
|
|
|
|
tls: DefaultTLSOptions,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"empty-TLS-values": func(t *testing.T) ConfigValidateTest {
|
|
|
|
return ConfigValidateTest{
|
|
|
|
config: &Config{
|
|
|
|
Address: "127.0.0.1",
|
|
|
|
Root: "testdata/secrets/root_ca.crt",
|
|
|
|
IntermediateCert: "testdata/secrets/intermediate_ca.crt",
|
|
|
|
IntermediateKey: "testdata/secrets/intermediate_ca_key",
|
|
|
|
DNSNames: []string{"test.smallstep.com"},
|
|
|
|
Password: "pass",
|
|
|
|
AuthorityConfig: ac,
|
|
|
|
TLS: &tlsutil.TLSOptions{},
|
|
|
|
},
|
|
|
|
tls: DefaultTLSOptions,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"custom-tls-values": func(t *testing.T) ConfigValidateTest {
|
|
|
|
return ConfigValidateTest{
|
|
|
|
config: &Config{
|
|
|
|
Address: "127.0.0.1",
|
|
|
|
Root: "testdata/secrets/root_ca.crt",
|
|
|
|
IntermediateCert: "testdata/secrets/intermediate_ca.crt",
|
|
|
|
IntermediateKey: "testdata/secrets/intermediate_ca_key",
|
|
|
|
DNSNames: []string{"test.smallstep.com"},
|
|
|
|
Password: "pass",
|
|
|
|
AuthorityConfig: ac,
|
|
|
|
TLS: &tlsutil.TLSOptions{
|
|
|
|
CipherSuites: x509util.CipherSuites{
|
|
|
|
"TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305",
|
|
|
|
},
|
|
|
|
MinVersion: 1.0,
|
|
|
|
MaxVersion: 1.1,
|
|
|
|
Renegotiation: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
tls: tlsutil.TLSOptions{
|
|
|
|
CipherSuites: x509util.CipherSuites{
|
|
|
|
"TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305",
|
|
|
|
},
|
|
|
|
MinVersion: 1.0,
|
|
|
|
MaxVersion: 1.1,
|
|
|
|
Renegotiation: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"tls-min>max": func(t *testing.T) ConfigValidateTest {
|
|
|
|
return ConfigValidateTest{
|
|
|
|
config: &Config{
|
|
|
|
Address: "127.0.0.1",
|
|
|
|
Root: "testdata/secrets/root_ca.crt",
|
|
|
|
IntermediateCert: "testdata/secrets/intermediate_ca.crt",
|
|
|
|
IntermediateKey: "testdata/secrets/intermediate_ca_key",
|
|
|
|
DNSNames: []string{"test.smallstep.com"},
|
|
|
|
Password: "pass",
|
|
|
|
AuthorityConfig: ac,
|
|
|
|
TLS: &tlsutil.TLSOptions{
|
|
|
|
CipherSuites: x509util.CipherSuites{
|
|
|
|
"TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305",
|
|
|
|
},
|
|
|
|
MinVersion: 1.2,
|
|
|
|
MaxVersion: 1.1,
|
|
|
|
Renegotiation: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
err: errors.New("tls minVersion cannot exceed tls maxVersion"),
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for name, get := range tests {
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
tc := get(t)
|
|
|
|
err := tc.config.Validate()
|
|
|
|
if err != nil {
|
|
|
|
if assert.NotNil(t, tc.err) {
|
|
|
|
assert.Equals(t, tc.err.Error(), err.Error())
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if assert.Nil(t, tc.err) {
|
|
|
|
assert.Equals(t, *tc.config.TLS, tc.tls)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAuthConfigValidate(t *testing.T) {
|
|
|
|
asn1dn := x509util.ASN1DN{
|
|
|
|
Country: "Tazmania",
|
|
|
|
Organization: "Acme Co",
|
|
|
|
Locality: "Landscapes",
|
|
|
|
Province: "Sudden Cliffs",
|
|
|
|
StreetAddress: "TNT",
|
|
|
|
CommonName: "test",
|
|
|
|
}
|
|
|
|
|
|
|
|
maxjwk, err := stepJOSE.ParseKey("testdata/secrets/max_pub.jwk")
|
|
|
|
assert.FatalError(t, err)
|
|
|
|
clijwk, err := stepJOSE.ParseKey("testdata/secrets/step_cli_key_pub.jwk")
|
|
|
|
assert.FatalError(t, err)
|
2018-10-12 06:03:00 +00:00
|
|
|
p := []*provisioner.Provisioner{
|
2018-10-05 21:48:36 +00:00
|
|
|
{
|
|
|
|
Issuer: "Max",
|
|
|
|
Type: "JWK",
|
|
|
|
Key: maxjwk,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Issuer: "step-cli",
|
|
|
|
Type: "JWK",
|
|
|
|
Key: clijwk,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
type AuthConfigValidateTest struct {
|
|
|
|
ac *AuthConfig
|
|
|
|
asn1dn x509util.ASN1DN
|
|
|
|
err error
|
|
|
|
}
|
|
|
|
tests := map[string]func(*testing.T) AuthConfigValidateTest{
|
2018-10-09 06:25:18 +00:00
|
|
|
"fail-nil-authconfig": func(t *testing.T) AuthConfigValidateTest {
|
2018-10-05 21:48:36 +00:00
|
|
|
return AuthConfigValidateTest{
|
|
|
|
ac: nil,
|
|
|
|
err: errors.New("authority cannot be undefined"),
|
|
|
|
}
|
|
|
|
},
|
2018-10-09 06:25:18 +00:00
|
|
|
"fail-empty-provisioners": func(t *testing.T) AuthConfigValidateTest {
|
2018-10-05 21:48:36 +00:00
|
|
|
return AuthConfigValidateTest{
|
|
|
|
ac: &AuthConfig{},
|
|
|
|
err: errors.New("authority.provisioners cannot be empty"),
|
|
|
|
}
|
|
|
|
},
|
2018-10-09 06:25:18 +00:00
|
|
|
"fail-invalid-provisioners": func(t *testing.T) AuthConfigValidateTest {
|
|
|
|
return AuthConfigValidateTest{
|
|
|
|
ac: &AuthConfig{
|
2018-10-12 06:03:00 +00:00
|
|
|
Provisioners: []*provisioner.Provisioner{
|
|
|
|
{Issuer: "foo", Type: "bar", Key: &jose.JSONWebKey{}},
|
|
|
|
{Issuer: "foo", Key: &jose.JSONWebKey{}},
|
2018-10-09 06:25:18 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
err: errors.New("provisioner type cannot be empty"),
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"ok-empty-asn1dn-template": func(t *testing.T) AuthConfigValidateTest {
|
2018-10-05 21:48:36 +00:00
|
|
|
return AuthConfigValidateTest{
|
|
|
|
ac: &AuthConfig{
|
|
|
|
Provisioners: p,
|
|
|
|
},
|
|
|
|
asn1dn: x509util.ASN1DN{},
|
|
|
|
}
|
|
|
|
},
|
2018-10-09 06:25:18 +00:00
|
|
|
"ok-custom-asn1dn": func(t *testing.T) AuthConfigValidateTest {
|
2018-10-05 21:48:36 +00:00
|
|
|
return AuthConfigValidateTest{
|
|
|
|
ac: &AuthConfig{
|
|
|
|
Provisioners: p,
|
|
|
|
Template: &asn1dn,
|
|
|
|
},
|
|
|
|
asn1dn: asn1dn,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for name, get := range tests {
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
tc := get(t)
|
|
|
|
err := tc.ac.Validate()
|
|
|
|
if err != nil {
|
|
|
|
if assert.NotNil(t, tc.err) {
|
|
|
|
assert.Equals(t, tc.err.Error(), err.Error())
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if assert.Nil(t, tc.err) {
|
|
|
|
assert.Equals(t, *tc.ac.Template, tc.asn1dn)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|