Implementation of serving not ready endpoints
This commit is contained in:
parent
a4c0b1649d
commit
9588e51146
13 changed files with 204 additions and 42 deletions
|
@ -13,9 +13,11 @@ import (
|
|||
"github.com/traefik/traefik/v3/pkg/logs"
|
||||
"github.com/traefik/traefik/v3/pkg/provider"
|
||||
traefikv1alpha1 "github.com/traefik/traefik/v3/pkg/provider/kubernetes/crd/traefikio/v1alpha1"
|
||||
"github.com/traefik/traefik/v3/pkg/provider/kubernetes/k8s"
|
||||
"github.com/traefik/traefik/v3/pkg/tls"
|
||||
corev1 "k8s.io/api/core/v1"
|
||||
"k8s.io/apimachinery/pkg/util/intstr"
|
||||
"k8s.io/utils/ptr"
|
||||
)
|
||||
|
||||
const (
|
||||
|
@ -544,7 +546,7 @@ func (c configBuilder) loadServers(parentNamespace string, svc traefikv1alpha1.L
|
|||
}
|
||||
|
||||
for _, endpoint := range endpointSlice.Endpoints {
|
||||
if endpoint.Conditions.Ready == nil || !*endpoint.Conditions.Ready {
|
||||
if !k8s.EndpointServing(endpoint) {
|
||||
continue
|
||||
}
|
||||
|
||||
|
@ -555,7 +557,8 @@ func (c configBuilder) loadServers(parentNamespace string, svc traefikv1alpha1.L
|
|||
|
||||
addresses[address] = struct{}{}
|
||||
servers = append(servers, dynamic.Server{
|
||||
URL: fmt.Sprintf("%s://%s", protocol, net.JoinHostPort(address, strconv.Itoa(int(port)))),
|
||||
URL: fmt.Sprintf("%s://%s", protocol, net.JoinHostPort(address, strconv.Itoa(int(port)))),
|
||||
Fenced: ptr.Deref(endpoint.Conditions.Serving, false),
|
||||
})
|
||||
}
|
||||
}
|
||||
|
|
|
@ -4737,6 +4737,14 @@ func TestLoadIngressRoutes(t *testing.T) {
|
|||
{
|
||||
URL: "http://10.10.0.2:80",
|
||||
},
|
||||
{
|
||||
URL: "http://10.10.0.3:80",
|
||||
Fenced: true,
|
||||
},
|
||||
{
|
||||
URL: "http://10.10.0.4:80",
|
||||
Fenced: true,
|
||||
},
|
||||
{
|
||||
URL: "http://10.10.0.5:80",
|
||||
},
|
||||
|
|
|
@ -18,7 +18,7 @@ func Test_convertSlice_corev1_to_networkingv1(t *testing.T) {
|
|||
{
|
||||
Port: 123,
|
||||
Protocol: "https",
|
||||
Error: ptr("test"),
|
||||
Error: pointer("test"),
|
||||
},
|
||||
},
|
||||
},
|
||||
|
@ -35,7 +35,7 @@ func Test_convertSlice_corev1_to_networkingv1(t *testing.T) {
|
|||
{
|
||||
Port: 123,
|
||||
Protocol: "https",
|
||||
Error: ptr("test"),
|
||||
Error: pointer("test"),
|
||||
},
|
||||
},
|
||||
},
|
||||
|
@ -52,7 +52,7 @@ func Test_convert(t *testing.T) {
|
|||
{
|
||||
Port: 123,
|
||||
Protocol: "https",
|
||||
Error: ptr("test"),
|
||||
Error: pointer("test"),
|
||||
},
|
||||
},
|
||||
}
|
||||
|
@ -67,14 +67,10 @@ func Test_convert(t *testing.T) {
|
|||
{
|
||||
Port: 123,
|
||||
Protocol: "https",
|
||||
Error: ptr("test"),
|
||||
Error: pointer("test"),
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
assert.Equal(t, expected, actual)
|
||||
}
|
||||
|
||||
func ptr[T any](v T) *T {
|
||||
return &v
|
||||
}
|
||||
|
|
|
@ -30,6 +30,7 @@ import (
|
|||
corev1 "k8s.io/api/core/v1"
|
||||
netv1 "k8s.io/api/networking/v1"
|
||||
"k8s.io/apimachinery/pkg/labels"
|
||||
"k8s.io/utils/ptr"
|
||||
)
|
||||
|
||||
const (
|
||||
|
@ -587,7 +588,7 @@ func (p *Provider) loadService(client Client, namespace string, backend netv1.In
|
|||
protocol := getProtocol(portSpec, portName, svcConfig)
|
||||
|
||||
for _, endpoint := range endpointSlice.Endpoints {
|
||||
if endpoint.Conditions.Ready == nil || !*endpoint.Conditions.Ready {
|
||||
if !k8s.EndpointServing(endpoint) {
|
||||
continue
|
||||
}
|
||||
|
||||
|
@ -598,7 +599,8 @@ func (p *Provider) loadService(client Client, namespace string, backend netv1.In
|
|||
|
||||
addresses[address] = struct{}{}
|
||||
svc.LoadBalancer.Servers = append(svc.LoadBalancer.Servers, dynamic.Server{
|
||||
URL: fmt.Sprintf("%s://%s", protocol, net.JoinHostPort(address, strconv.Itoa(int(port)))),
|
||||
URL: fmt.Sprintf("%s://%s", protocol, net.JoinHostPort(address, strconv.Itoa(int(port)))),
|
||||
Fenced: ptr.Deref(endpoint.Conditions.Serving, false),
|
||||
})
|
||||
}
|
||||
}
|
||||
|
|
11
pkg/provider/kubernetes/k8s/endpoint.go
Normal file
11
pkg/provider/kubernetes/k8s/endpoint.go
Normal file
|
@ -0,0 +1,11 @@
|
|||
package k8s
|
||||
|
||||
import (
|
||||
v1 "k8s.io/api/discovery/v1"
|
||||
"k8s.io/utils/ptr"
|
||||
)
|
||||
|
||||
// EndpointServing returns true if the endpoint is still serving the service.
|
||||
func EndpointServing(endpoint v1.Endpoint) bool {
|
||||
return ptr.Deref(endpoint.Conditions.Ready, false) || ptr.Deref(endpoint.Conditions.Serving, false)
|
||||
}
|
75
pkg/provider/kubernetes/k8s/endpoint_test.go
Normal file
75
pkg/provider/kubernetes/k8s/endpoint_test.go
Normal file
|
@ -0,0 +1,75 @@
|
|||
package k8s
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/assert"
|
||||
v1 "k8s.io/api/discovery/v1"
|
||||
)
|
||||
|
||||
func TestEndpointServing(t *testing.T) {
|
||||
tests := []struct {
|
||||
name string
|
||||
endpoint v1.Endpoint
|
||||
want bool
|
||||
}{
|
||||
{
|
||||
name: "no status",
|
||||
endpoint: v1.Endpoint{
|
||||
Conditions: v1.EndpointConditions{
|
||||
Ready: nil,
|
||||
Serving: nil,
|
||||
},
|
||||
},
|
||||
want: false,
|
||||
},
|
||||
{
|
||||
name: "ready",
|
||||
endpoint: v1.Endpoint{
|
||||
Conditions: v1.EndpointConditions{
|
||||
Ready: pointer(true),
|
||||
Serving: nil,
|
||||
},
|
||||
},
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "not ready",
|
||||
endpoint: v1.Endpoint{
|
||||
Conditions: v1.EndpointConditions{
|
||||
Ready: pointer(false),
|
||||
Serving: nil,
|
||||
},
|
||||
},
|
||||
want: false,
|
||||
},
|
||||
{
|
||||
name: "not ready and serving",
|
||||
endpoint: v1.Endpoint{
|
||||
Conditions: v1.EndpointConditions{
|
||||
Ready: pointer(false),
|
||||
Serving: pointer(true),
|
||||
},
|
||||
},
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "not ready and not serving",
|
||||
endpoint: v1.Endpoint{
|
||||
Conditions: v1.EndpointConditions{
|
||||
Ready: pointer(false),
|
||||
Serving: pointer(false),
|
||||
},
|
||||
},
|
||||
want: false,
|
||||
},
|
||||
}
|
||||
for _, test := range tests {
|
||||
t.Run(test.name, func(t *testing.T) {
|
||||
got := EndpointServing(test.endpoint)
|
||||
assert.Equal(t, test.want, got)
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func pointer[T any](v T) *T { return &v }
|
Loading…
Add table
Add a link
Reference in a new issue