mirror of https://github.com/hashicorp/consul
structs: add some missing config entry validation and clean up tests (#10465)
Affects kinds: service-defaults, ingress-gateway, terminating-gatewaypull/10474/head
parent
77a3432f76
commit
ac50db9087
|
@ -6,16 +6,12 @@ import (
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestIngressConfigEntry_Normalize(t *testing.T) {
|
func TestIngressGatewayConfigEntry(t *testing.T) {
|
||||||
|
defaultMeta := DefaultEnterpriseMeta()
|
||||||
|
|
||||||
cases := []struct {
|
cases := map[string]configEntryTestcase{
|
||||||
name string
|
"normalize: empty protocol": {
|
||||||
entry IngressGatewayConfigEntry
|
entry: &IngressGatewayConfigEntry{
|
||||||
expected IngressGatewayConfigEntry
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
name: "empty protocol",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
Kind: "ingress-gateway",
|
||||||
Name: "ingress-web",
|
Name: "ingress-web",
|
||||||
Listeners: []IngressListener{
|
Listeners: []IngressListener{
|
||||||
|
@ -26,7 +22,7 @@ func TestIngressConfigEntry_Normalize(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
expected: IngressGatewayConfigEntry{
|
expected: &IngressGatewayConfigEntry{
|
||||||
Kind: "ingress-gateway",
|
Kind: "ingress-gateway",
|
||||||
Name: "ingress-web",
|
Name: "ingress-web",
|
||||||
Listeners: []IngressListener{
|
Listeners: []IngressListener{
|
||||||
|
@ -36,12 +32,12 @@ func TestIngressConfigEntry_Normalize(t *testing.T) {
|
||||||
Services: []IngressService{},
|
Services: []IngressService{},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
EnterpriseMeta: *defaultMeta,
|
||||||
},
|
},
|
||||||
|
normalizeOnly: true,
|
||||||
},
|
},
|
||||||
{
|
"normalize: lowercase protocols": {
|
||||||
name: "lowercase protocols",
|
entry: &IngressGatewayConfigEntry{
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
Kind: "ingress-gateway",
|
||||||
Name: "ingress-web",
|
Name: "ingress-web",
|
||||||
Listeners: []IngressListener{
|
Listeners: []IngressListener{
|
||||||
|
@ -57,7 +53,7 @@ func TestIngressConfigEntry_Normalize(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
expected: IngressGatewayConfigEntry{
|
expected: &IngressGatewayConfigEntry{
|
||||||
Kind: "ingress-gateway",
|
Kind: "ingress-gateway",
|
||||||
Name: "ingress-web",
|
Name: "ingress-web",
|
||||||
Listeners: []IngressListener{
|
Listeners: []IngressListener{
|
||||||
|
@ -72,18 +68,335 @@ func TestIngressConfigEntry_Normalize(t *testing.T) {
|
||||||
Services: []IngressService{},
|
Services: []IngressService{},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
EnterpriseMeta: *defaultMeta,
|
||||||
},
|
},
|
||||||
|
normalizeOnly: true,
|
||||||
|
},
|
||||||
|
"port conflict": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "tcp",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "mysql",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "tcp",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "postgres",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: "port 1111 declared on two listeners",
|
||||||
|
},
|
||||||
|
"http features: wildcard": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "http",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "*",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"http features: wildcard service on invalid protocol": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "tcp",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "*",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: "Wildcard service name is only valid for protocol",
|
||||||
|
},
|
||||||
|
"http features: multiple services": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "tcp",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "db1",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "db2",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: "Multiple services per listener are only supported for protocol",
|
||||||
|
},
|
||||||
|
// ==========================
|
||||||
|
"tcp listener requires a defined service": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "tcp",
|
||||||
|
Services: []IngressService{},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: "No service declared for listener with port 1111",
|
||||||
|
},
|
||||||
|
"http listener requires a defined service": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "http",
|
||||||
|
Services: []IngressService{},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: "No service declared for listener with port 1111",
|
||||||
|
},
|
||||||
|
"empty service name not supported": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "tcp",
|
||||||
|
Services: []IngressService{
|
||||||
|
{},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: "Service name cannot be blank",
|
||||||
|
},
|
||||||
|
"protocol validation": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "asdf",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "db",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: "protocol must be 'tcp', 'http', 'http2', or 'grpc'. 'asdf' is an unsupported protocol",
|
||||||
|
},
|
||||||
|
"hosts cannot be set on a tcp listener": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "tcp",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "db",
|
||||||
|
Hosts: []string{"db.example.com"},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: "Associating hosts to a service is not supported for the tcp protocol",
|
||||||
|
},
|
||||||
|
"hosts cannot be set on a wildcard specifier": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "http",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "*",
|
||||||
|
Hosts: []string{"db.example.com"},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: "Associating hosts to a wildcard service is not supported",
|
||||||
|
},
|
||||||
|
"hosts must be unique per listener": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "http",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "db",
|
||||||
|
Hosts: []string{"test.example.com"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "api",
|
||||||
|
Hosts: []string{"test.example.com"},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: "Hosts must be unique within a specific listener",
|
||||||
|
},
|
||||||
|
"hosts must be a valid DNS name": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "http",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "db",
|
||||||
|
Hosts: []string{"example..com"},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: `Host "example..com" must be a valid DNS hostname`,
|
||||||
|
},
|
||||||
|
"wildcard specifier is only allowed in the leftmost label": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "http",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "db",
|
||||||
|
Hosts: []string{"*.example.com"},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"wildcard specifier is not allowed in non-leftmost labels": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "http",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "db",
|
||||||
|
Hosts: []string{"example.*.com"},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: `Host "example.*.com" is not valid, a wildcard specifier is only allowed as the leftmost label`,
|
||||||
|
},
|
||||||
|
"wildcard specifier is not allowed in leftmost labels as a partial": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "http",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "db",
|
||||||
|
Hosts: []string{"*-test.example.com"},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: `Host "*-test.example.com" is not valid, a wildcard specifier is only allowed as the leftmost label`,
|
||||||
|
},
|
||||||
|
"wildcard specifier is allowed for hosts when TLS is disabled": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "http",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "db",
|
||||||
|
Hosts: []string{"*"},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"wildcard specifier is not allowed for hosts when TLS is enabled": {
|
||||||
|
entry: &IngressGatewayConfigEntry{
|
||||||
|
Kind: "ingress-gateway",
|
||||||
|
Name: "ingress-web",
|
||||||
|
TLS: GatewayTLSConfig{
|
||||||
|
Enabled: true,
|
||||||
|
},
|
||||||
|
Listeners: []IngressListener{
|
||||||
|
{
|
||||||
|
Port: 1111,
|
||||||
|
Protocol: "http",
|
||||||
|
Services: []IngressService{
|
||||||
|
{
|
||||||
|
Name: "db",
|
||||||
|
Hosts: []string{"*"},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
validateErr: `Host '*' is not allowed when TLS is enabled, all hosts must be valid DNS records to add as a DNSSAN`,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tc := range cases {
|
testConfigEntryNormalizeAndValidate(t, cases)
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
|
||||||
err := tc.entry.Normalize()
|
|
||||||
require.NoError(t, err)
|
|
||||||
require.Equal(t, tc.expected, tc.entry)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestIngressConfigEntry_ListRelatedServices(t *testing.T) {
|
func TestIngressConfigEntry_ListRelatedServices(t *testing.T) {
|
||||||
|
@ -168,376 +481,10 @@ func TestIngressConfigEntry_ListRelatedServices(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestIngressConfigEntry_Validate(t *testing.T) {
|
func TestTerminatingGatewayConfigEntry(t *testing.T) {
|
||||||
|
cases := map[string]configEntryTestcase{
|
||||||
cases := []struct {
|
"service conflict": {
|
||||||
name string
|
entry: &TerminatingGatewayConfigEntry{
|
||||||
entry IngressGatewayConfigEntry
|
|
||||||
expectErr string
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
name: "port conflict",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "tcp",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "mysql",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "tcp",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "postgres",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: "port 1111 declared on two listeners",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "http features: wildcard",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "http",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "*",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "http features: wildcard service on invalid protocol",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "tcp",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "*",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: "Wildcard service name is only valid for protocol",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "http features: multiple services",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "tcp",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "db1",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Name: "db2",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: "multiple services per listener are only supported for protocol",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "tcp listener requires a defined service",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "tcp",
|
|
||||||
Services: []IngressService{},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: "no service declared for listener with port 1111",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "http listener requires a defined service",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "http",
|
|
||||||
Services: []IngressService{},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: "no service declared for listener with port 1111",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "empty service name not supported",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "tcp",
|
|
||||||
Services: []IngressService{
|
|
||||||
{},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: "Service name cannot be blank",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "protocol validation",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "asdf",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "db",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: "protocol must be 'tcp', 'http', 'http2', or 'grpc'. 'asdf' is an unsupported protocol",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "hosts cannot be set on a tcp listener",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "tcp",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "db",
|
|
||||||
Hosts: []string{"db.example.com"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: "Associating hosts to a service is not supported for the tcp protocol",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "hosts cannot be set on a wildcard specifier",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "http",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "*",
|
|
||||||
Hosts: []string{"db.example.com"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: "Associating hosts to a wildcard service is not supported",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "hosts must be unique per listener",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "http",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "db",
|
|
||||||
Hosts: []string{"test.example.com"},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Name: "api",
|
|
||||||
Hosts: []string{"test.example.com"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: "Hosts must be unique within a specific listener",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "hosts must be a valid DNS name",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "http",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "db",
|
|
||||||
Hosts: []string{"example..com"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: `Host "example..com" must be a valid DNS hostname`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "wildcard specifier is only allowed in the leftmost label",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "http",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "db",
|
|
||||||
Hosts: []string{"*.example.com"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "wildcard specifier is not allowed in non-leftmost labels",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "http",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "db",
|
|
||||||
Hosts: []string{"example.*.com"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: `Host "example.*.com" is not valid, a wildcard specifier is only allowed as the leftmost label`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "wildcard specifier is not allowed in leftmost labels as a partial",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "http",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "db",
|
|
||||||
Hosts: []string{"*-test.example.com"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: `Host "*-test.example.com" is not valid, a wildcard specifier is only allowed as the leftmost label`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "wildcard specifier is allowed for hosts when TLS is disabled",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "http",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "db",
|
|
||||||
Hosts: []string{"*"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "wildcard specifier is not allowed for hosts when TLS is enabled",
|
|
||||||
entry: IngressGatewayConfigEntry{
|
|
||||||
Kind: "ingress-gateway",
|
|
||||||
Name: "ingress-web",
|
|
||||||
TLS: GatewayTLSConfig{
|
|
||||||
Enabled: true,
|
|
||||||
},
|
|
||||||
Listeners: []IngressListener{
|
|
||||||
{
|
|
||||||
Port: 1111,
|
|
||||||
Protocol: "http",
|
|
||||||
Services: []IngressService{
|
|
||||||
{
|
|
||||||
Name: "db",
|
|
||||||
Hosts: []string{"*"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expectErr: `Host '*' is not allowed when TLS is enabled, all hosts must be valid DNS records to add as a DNSSAN`,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, tc := range cases {
|
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
|
||||||
err := tc.entry.Validate()
|
|
||||||
if tc.expectErr != "" {
|
|
||||||
require.Error(t, err)
|
|
||||||
requireContainsLower(t, err.Error(), tc.expectErr)
|
|
||||||
} else {
|
|
||||||
require.NoError(t, err)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTerminatingConfigEntry_Validate(t *testing.T) {
|
|
||||||
|
|
||||||
cases := []struct {
|
|
||||||
name string
|
|
||||||
entry TerminatingGatewayConfigEntry
|
|
||||||
expectErr string
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
name: "service conflict",
|
|
||||||
entry: TerminatingGatewayConfigEntry{
|
|
||||||
Kind: "terminating-gateway",
|
Kind: "terminating-gateway",
|
||||||
Name: "terminating-gw-west",
|
Name: "terminating-gw-west",
|
||||||
Services: []LinkedService{
|
Services: []LinkedService{
|
||||||
|
@ -549,11 +496,10 @@ func TestTerminatingConfigEntry_Validate(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
expectErr: "specified more than once",
|
validateErr: "specified more than once",
|
||||||
},
|
},
|
||||||
{
|
"blank service name": {
|
||||||
name: "blank service name",
|
entry: &TerminatingGatewayConfigEntry{
|
||||||
entry: TerminatingGatewayConfigEntry{
|
|
||||||
Kind: "terminating-gateway",
|
Kind: "terminating-gateway",
|
||||||
Name: "terminating-gw-west",
|
Name: "terminating-gw-west",
|
||||||
Services: []LinkedService{
|
Services: []LinkedService{
|
||||||
|
@ -562,11 +508,10 @@ func TestTerminatingConfigEntry_Validate(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
expectErr: "Service name cannot be blank.",
|
validateErr: "Service name cannot be blank.",
|
||||||
},
|
},
|
||||||
{
|
"not all TLS options provided-1": {
|
||||||
name: "not all TLS options provided-1",
|
entry: &TerminatingGatewayConfigEntry{
|
||||||
entry: TerminatingGatewayConfigEntry{
|
|
||||||
Kind: "terminating-gateway",
|
Kind: "terminating-gateway",
|
||||||
Name: "terminating-gw-west",
|
Name: "terminating-gw-west",
|
||||||
Services: []LinkedService{
|
Services: []LinkedService{
|
||||||
|
@ -576,11 +521,10 @@ func TestTerminatingConfigEntry_Validate(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
expectErr: "must have a CertFile, CAFile, and KeyFile",
|
validateErr: "must have a CertFile, CAFile, and KeyFile",
|
||||||
},
|
},
|
||||||
{
|
"not all TLS options provided-2": {
|
||||||
name: "not all TLS options provided-2",
|
entry: &TerminatingGatewayConfigEntry{
|
||||||
entry: TerminatingGatewayConfigEntry{
|
|
||||||
Kind: "terminating-gateway",
|
Kind: "terminating-gateway",
|
||||||
Name: "terminating-gw-west",
|
Name: "terminating-gw-west",
|
||||||
Services: []LinkedService{
|
Services: []LinkedService{
|
||||||
|
@ -590,11 +534,10 @@ func TestTerminatingConfigEntry_Validate(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
expectErr: "must have a CertFile, CAFile, and KeyFile",
|
validateErr: "must have a CertFile, CAFile, and KeyFile",
|
||||||
},
|
},
|
||||||
{
|
"all TLS options provided": {
|
||||||
name: "all TLS options provided",
|
entry: &TerminatingGatewayConfigEntry{
|
||||||
entry: TerminatingGatewayConfigEntry{
|
|
||||||
Kind: "terminating-gateway",
|
Kind: "terminating-gateway",
|
||||||
Name: "terminating-gw-west",
|
Name: "terminating-gw-west",
|
||||||
Services: []LinkedService{
|
Services: []LinkedService{
|
||||||
|
@ -607,9 +550,8 @@ func TestTerminatingConfigEntry_Validate(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
"only providing ca file is allowed": {
|
||||||
name: "only providing ca file is allowed",
|
entry: &TerminatingGatewayConfigEntry{
|
||||||
entry: TerminatingGatewayConfigEntry{
|
|
||||||
Kind: "terminating-gateway",
|
Kind: "terminating-gateway",
|
||||||
Name: "terminating-gw-west",
|
Name: "terminating-gw-west",
|
||||||
Services: []LinkedService{
|
Services: []LinkedService{
|
||||||
|
@ -621,19 +563,7 @@ func TestTerminatingConfigEntry_Validate(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
testConfigEntryNormalizeAndValidate(t, cases)
|
||||||
for _, tc := range cases {
|
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
|
||||||
|
|
||||||
err := tc.entry.Validate()
|
|
||||||
if tc.expectErr != "" {
|
|
||||||
require.Error(t, err)
|
|
||||||
requireContainsLower(t, err.Error(), tc.expectErr)
|
|
||||||
} else {
|
|
||||||
require.NoError(t, err)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGatewayService_Addresses(t *testing.T) {
|
func TestGatewayService_Addresses(t *testing.T) {
|
||||||
|
|
|
@ -1646,16 +1646,11 @@ func TestUpstreamLimits_Validate(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestServiceConfigEntry_Normalize(t *testing.T) {
|
func TestServiceConfigEntry(t *testing.T) {
|
||||||
tt := []struct {
|
cases := map[string]configEntryTestcase{
|
||||||
name string
|
"normalize: upstream config override no name": {
|
||||||
input ServiceConfigEntry
|
|
||||||
expect ServiceConfigEntry
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
// This will do nothing to normalization, but it will fail at validation later
|
// This will do nothing to normalization, but it will fail at validation later
|
||||||
name: "upstream config override no name",
|
entry: &ServiceConfigEntry{
|
||||||
input: ServiceConfigEntry{
|
|
||||||
Name: "web",
|
Name: "web",
|
||||||
UpstreamConfig: &UpstreamConfiguration{
|
UpstreamConfig: &UpstreamConfiguration{
|
||||||
Overrides: []*UpstreamConfig{
|
Overrides: []*UpstreamConfig{
|
||||||
|
@ -1673,7 +1668,7 @@ func TestServiceConfigEntry_Normalize(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
expect: ServiceConfigEntry{
|
expected: &ServiceConfigEntry{
|
||||||
Kind: ServiceDefaults,
|
Kind: ServiceDefaults,
|
||||||
Name: "web",
|
Name: "web",
|
||||||
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
||||||
|
@ -1696,11 +1691,11 @@ func TestServiceConfigEntry_Normalize(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
normalizeOnly: true,
|
||||||
},
|
},
|
||||||
{
|
"normalize: upstream config defaults with name": {
|
||||||
// This will do nothing to normalization, but it will fail at validation later
|
// This will do nothing to normalization, but it will fail at validation later
|
||||||
name: "upstream config defaults with name",
|
entry: &ServiceConfigEntry{
|
||||||
input: ServiceConfigEntry{
|
|
||||||
Name: "web",
|
Name: "web",
|
||||||
UpstreamConfig: &UpstreamConfiguration{
|
UpstreamConfig: &UpstreamConfiguration{
|
||||||
Defaults: &UpstreamConfig{
|
Defaults: &UpstreamConfig{
|
||||||
|
@ -1709,7 +1704,7 @@ func TestServiceConfigEntry_Normalize(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
expect: ServiceConfigEntry{
|
expected: &ServiceConfigEntry{
|
||||||
Kind: ServiceDefaults,
|
Kind: ServiceDefaults,
|
||||||
Name: "web",
|
Name: "web",
|
||||||
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
||||||
|
@ -1720,35 +1715,35 @@ func TestServiceConfigEntry_Normalize(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
normalizeOnly: true,
|
||||||
},
|
},
|
||||||
{
|
"normalize: fill-in-kind": {
|
||||||
name: "fill-in-kind",
|
entry: &ServiceConfigEntry{
|
||||||
input: ServiceConfigEntry{
|
|
||||||
Name: "web",
|
Name: "web",
|
||||||
},
|
},
|
||||||
expect: ServiceConfigEntry{
|
expected: &ServiceConfigEntry{
|
||||||
Kind: ServiceDefaults,
|
Kind: ServiceDefaults,
|
||||||
Name: "web",
|
Name: "web",
|
||||||
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
||||||
},
|
},
|
||||||
|
normalizeOnly: true,
|
||||||
},
|
},
|
||||||
{
|
"normalize: lowercase-protocol": {
|
||||||
name: "lowercase-protocol",
|
entry: &ServiceConfigEntry{
|
||||||
input: ServiceConfigEntry{
|
|
||||||
Kind: ServiceDefaults,
|
Kind: ServiceDefaults,
|
||||||
Name: "web",
|
Name: "web",
|
||||||
Protocol: "PrOtoCoL",
|
Protocol: "PrOtoCoL",
|
||||||
},
|
},
|
||||||
expect: ServiceConfigEntry{
|
expected: &ServiceConfigEntry{
|
||||||
Kind: ServiceDefaults,
|
Kind: ServiceDefaults,
|
||||||
Name: "web",
|
Name: "web",
|
||||||
Protocol: "protocol",
|
Protocol: "protocol",
|
||||||
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
||||||
},
|
},
|
||||||
|
normalizeOnly: true,
|
||||||
},
|
},
|
||||||
{
|
"normalize: connect-kitchen-sink": {
|
||||||
name: "connect-kitchen-sink",
|
entry: &ServiceConfigEntry{
|
||||||
input: ServiceConfigEntry{
|
|
||||||
Kind: ServiceDefaults,
|
Kind: ServiceDefaults,
|
||||||
Name: "web",
|
Name: "web",
|
||||||
UpstreamConfig: &UpstreamConfiguration{
|
UpstreamConfig: &UpstreamConfiguration{
|
||||||
|
@ -1766,7 +1761,7 @@ func TestServiceConfigEntry_Normalize(t *testing.T) {
|
||||||
},
|
},
|
||||||
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
||||||
},
|
},
|
||||||
expect: ServiceConfigEntry{
|
expected: &ServiceConfigEntry{
|
||||||
Kind: ServiceDefaults,
|
Kind: ServiceDefaults,
|
||||||
Name: "web",
|
Name: "web",
|
||||||
UpstreamConfig: &UpstreamConfiguration{
|
UpstreamConfig: &UpstreamConfiguration{
|
||||||
|
@ -1789,35 +1784,16 @@ func TestServiceConfigEntry_Normalize(t *testing.T) {
|
||||||
},
|
},
|
||||||
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
||||||
},
|
},
|
||||||
|
normalizeOnly: true,
|
||||||
},
|
},
|
||||||
}
|
"wildcard name is not allowed": {
|
||||||
|
entry: &ServiceConfigEntry{
|
||||||
for _, tc := range tt {
|
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
|
||||||
err := tc.input.Normalize()
|
|
||||||
require.NoError(t, err)
|
|
||||||
require.Equal(t, tc.expect, tc.input)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestServiceConfigEntry_Validate(t *testing.T) {
|
|
||||||
tt := []struct {
|
|
||||||
name string
|
|
||||||
input *ServiceConfigEntry
|
|
||||||
expect *ServiceConfigEntry
|
|
||||||
expectErr string
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
name: "wildcard name is not allowed",
|
|
||||||
input: &ServiceConfigEntry{
|
|
||||||
Name: WildcardSpecifier,
|
Name: WildcardSpecifier,
|
||||||
},
|
},
|
||||||
expectErr: `must be the name of a service, and not a wildcard`,
|
validateErr: `must be the name of a service, and not a wildcard`,
|
||||||
},
|
},
|
||||||
{
|
"upstream config override no name": {
|
||||||
name: "upstream config override no name",
|
entry: &ServiceConfigEntry{
|
||||||
input: &ServiceConfigEntry{
|
|
||||||
Name: "web",
|
Name: "web",
|
||||||
UpstreamConfig: &UpstreamConfiguration{
|
UpstreamConfig: &UpstreamConfiguration{
|
||||||
Overrides: []*UpstreamConfig{
|
Overrides: []*UpstreamConfig{
|
||||||
|
@ -1835,11 +1811,10 @@ func TestServiceConfigEntry_Validate(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
expectErr: `Name is required`,
|
validateErr: `Name is required`,
|
||||||
},
|
},
|
||||||
{
|
"upstream config defaults with name": {
|
||||||
name: "upstream config defaults with name",
|
entry: &ServiceConfigEntry{
|
||||||
input: &ServiceConfigEntry{
|
|
||||||
Name: "web",
|
Name: "web",
|
||||||
UpstreamConfig: &UpstreamConfiguration{
|
UpstreamConfig: &UpstreamConfiguration{
|
||||||
Defaults: &UpstreamConfig{
|
Defaults: &UpstreamConfig{
|
||||||
|
@ -1848,11 +1823,10 @@ func TestServiceConfigEntry_Validate(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
expectErr: `error in upstream defaults: Name must be empty`,
|
validateErr: `error in upstream defaults: Name must be empty`,
|
||||||
},
|
},
|
||||||
{
|
"connect-kitchen-sink": {
|
||||||
name: "connect-kitchen-sink",
|
entry: &ServiceConfigEntry{
|
||||||
input: &ServiceConfigEntry{
|
|
||||||
Kind: ServiceDefaults,
|
Kind: ServiceDefaults,
|
||||||
Name: "web",
|
Name: "web",
|
||||||
UpstreamConfig: &UpstreamConfiguration{
|
UpstreamConfig: &UpstreamConfiguration{
|
||||||
|
@ -1870,7 +1844,7 @@ func TestServiceConfigEntry_Validate(t *testing.T) {
|
||||||
},
|
},
|
||||||
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
EnterpriseMeta: *DefaultEnterpriseMeta(),
|
||||||
},
|
},
|
||||||
expect: &ServiceConfigEntry{
|
expected: &ServiceConfigEntry{
|
||||||
Kind: ServiceDefaults,
|
Kind: ServiceDefaults,
|
||||||
Name: "web",
|
Name: "web",
|
||||||
UpstreamConfig: &UpstreamConfiguration{
|
UpstreamConfig: &UpstreamConfiguration{
|
||||||
|
@ -1893,21 +1867,7 @@ func TestServiceConfigEntry_Validate(t *testing.T) {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
testConfigEntryNormalizeAndValidate(t, cases)
|
||||||
for _, tc := range tt {
|
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
|
||||||
// normalize before validate since they always happen in that order
|
|
||||||
require.NoError(t, tc.input.Normalize())
|
|
||||||
|
|
||||||
err := tc.input.Validate()
|
|
||||||
if tc.expectErr != "" {
|
|
||||||
testutil.RequireErrorContains(t, err, tc.expectErr)
|
|
||||||
} else {
|
|
||||||
require.NoError(t, err)
|
|
||||||
require.Equal(t, tc.expect, tc.input)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestUpstreamConfig_MergeInto(t *testing.T) {
|
func TestUpstreamConfig_MergeInto(t *testing.T) {
|
||||||
|
@ -2228,3 +2188,57 @@ func TestServiceConfigRequest_CacheInfoKey(t *testing.T) {
|
||||||
func TestDiscoveryChainRequest_CacheInfoKey(t *testing.T) {
|
func TestDiscoveryChainRequest_CacheInfoKey(t *testing.T) {
|
||||||
assertCacheInfoKeyIsComplete(t, &DiscoveryChainRequest{})
|
assertCacheInfoKeyIsComplete(t, &DiscoveryChainRequest{})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type configEntryTestcase struct {
|
||||||
|
entry ConfigEntry
|
||||||
|
normalizeOnly bool
|
||||||
|
|
||||||
|
normalizeErr string
|
||||||
|
validateErr string
|
||||||
|
|
||||||
|
// Only one of either expected or check can be set.
|
||||||
|
expected ConfigEntry
|
||||||
|
// check is called between normalize and validate
|
||||||
|
check func(t *testing.T, entry ConfigEntry)
|
||||||
|
}
|
||||||
|
|
||||||
|
func testConfigEntryNormalizeAndValidate(t *testing.T, cases map[string]configEntryTestcase) {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
for name, tc := range cases {
|
||||||
|
tc := tc
|
||||||
|
t.Run(name, func(t *testing.T) {
|
||||||
|
err := tc.entry.Normalize()
|
||||||
|
if tc.normalizeErr != "" {
|
||||||
|
testutil.RequireErrorContains(t, err, tc.normalizeErr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
if tc.expected != nil && tc.check != nil {
|
||||||
|
t.Fatal("cannot set both 'expected' and 'check' test case fields")
|
||||||
|
}
|
||||||
|
|
||||||
|
if tc.expected != nil {
|
||||||
|
require.Equal(t, tc.expected, tc.entry)
|
||||||
|
}
|
||||||
|
|
||||||
|
if tc.check != nil {
|
||||||
|
tc.check(t, tc.entry)
|
||||||
|
}
|
||||||
|
|
||||||
|
if tc.normalizeOnly {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
err = tc.entry.Validate()
|
||||||
|
if tc.validateErr != "" {
|
||||||
|
// require.Error(t, err)
|
||||||
|
// require.Contains(t, err.Error(), tc.validateErr)
|
||||||
|
testutil.RequireErrorContains(t, err, tc.validateErr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
require.NoError(t, err)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue