2020-12-15 16:40:05 +01:00
|
|
|
package gateway
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2021-03-04 20:08:03 +01:00
|
|
|
"os"
|
2020-12-15 16:40:05 +01:00
|
|
|
"path/filepath"
|
|
|
|
|
2023-02-03 15:24:05 +01:00
|
|
|
"github.com/traefik/traefik/v3/pkg/provider/kubernetes/k8s"
|
2020-12-15 16:40:05 +01:00
|
|
|
corev1 "k8s.io/api/core/v1"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
|
|
"k8s.io/apimachinery/pkg/labels"
|
2023-04-17 10:56:36 +02:00
|
|
|
kscheme "k8s.io/client-go/kubernetes/scheme"
|
2024-01-09 10:28:05 +01:00
|
|
|
gatev1 "sigs.k8s.io/gateway-api/apis/v1"
|
2023-04-17 10:56:36 +02:00
|
|
|
gatev1alpha2 "sigs.k8s.io/gateway-api/apis/v1alpha2"
|
2024-01-30 16:44:05 +01:00
|
|
|
gatev1beta1 "sigs.k8s.io/gateway-api/apis/v1beta1"
|
2020-12-15 16:40:05 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var _ Client = (*clientMock)(nil)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
// required by k8s.MustParseYaml
|
2023-04-17 10:56:36 +02:00
|
|
|
err := gatev1alpha2.AddToScheme(kscheme.Scheme)
|
2020-12-15 16:40:05 +01:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-01-09 10:28:05 +01:00
|
|
|
|
2024-01-30 16:44:05 +01:00
|
|
|
err = gatev1beta1.AddToScheme(kscheme.Scheme)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2024-01-09 10:28:05 +01:00
|
|
|
err = gatev1.AddToScheme(kscheme.Scheme)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2020-12-15 16:40:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type clientMock struct {
|
2021-10-04 15:46:08 +02:00
|
|
|
services []*corev1.Service
|
|
|
|
secrets []*corev1.Secret
|
|
|
|
endpoints []*corev1.Endpoints
|
|
|
|
namespaces []*corev1.Namespace
|
2020-12-15 16:40:05 +01:00
|
|
|
|
|
|
|
apiServiceError error
|
|
|
|
apiSecretError error
|
|
|
|
apiEndpointsError error
|
|
|
|
|
2024-01-30 16:44:05 +01:00
|
|
|
gatewayClasses []*gatev1.GatewayClass
|
|
|
|
gateways []*gatev1.Gateway
|
|
|
|
httpRoutes []*gatev1.HTTPRoute
|
|
|
|
tcpRoutes []*gatev1alpha2.TCPRoute
|
|
|
|
tlsRoutes []*gatev1alpha2.TLSRoute
|
|
|
|
referenceGrants []*gatev1beta1.ReferenceGrant
|
2020-12-15 16:40:05 +01:00
|
|
|
|
|
|
|
watchChan chan interface{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func newClientMock(paths ...string) clientMock {
|
|
|
|
var c clientMock
|
|
|
|
|
|
|
|
for _, path := range paths {
|
2021-03-04 20:08:03 +01:00
|
|
|
yamlContent, err := os.ReadFile(filepath.FromSlash("./fixtures/" + path))
|
2020-12-15 16:40:05 +01:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
k8sObjects := k8s.MustParseYaml(yamlContent)
|
|
|
|
for _, obj := range k8sObjects {
|
|
|
|
switch o := obj.(type) {
|
|
|
|
case *corev1.Service:
|
|
|
|
c.services = append(c.services, o)
|
|
|
|
case *corev1.Secret:
|
|
|
|
c.secrets = append(c.secrets, o)
|
2021-10-04 15:46:08 +02:00
|
|
|
case *corev1.Namespace:
|
|
|
|
c.namespaces = append(c.namespaces, o)
|
2020-12-15 16:40:05 +01:00
|
|
|
case *corev1.Endpoints:
|
|
|
|
c.endpoints = append(c.endpoints, o)
|
2024-01-09 10:28:05 +01:00
|
|
|
case *gatev1.GatewayClass:
|
2020-12-15 16:40:05 +01:00
|
|
|
c.gatewayClasses = append(c.gatewayClasses, o)
|
2024-01-09 10:28:05 +01:00
|
|
|
case *gatev1.Gateway:
|
2020-12-15 16:40:05 +01:00
|
|
|
c.gateways = append(c.gateways, o)
|
2024-01-09 10:28:05 +01:00
|
|
|
case *gatev1.HTTPRoute:
|
2020-12-15 16:40:05 +01:00
|
|
|
c.httpRoutes = append(c.httpRoutes, o)
|
2023-04-17 10:56:36 +02:00
|
|
|
case *gatev1alpha2.TCPRoute:
|
2021-05-20 11:50:12 +02:00
|
|
|
c.tcpRoutes = append(c.tcpRoutes, o)
|
2023-04-17 10:56:36 +02:00
|
|
|
case *gatev1alpha2.TLSRoute:
|
2021-05-20 11:50:12 +02:00
|
|
|
c.tlsRoutes = append(c.tlsRoutes, o)
|
2024-01-30 16:44:05 +01:00
|
|
|
case *gatev1beta1.ReferenceGrant:
|
|
|
|
c.referenceGrants = append(c.referenceGrants, o)
|
2020-12-15 16:40:05 +01:00
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("Unknown runtime object %+v %T", o, o))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
2024-01-09 10:28:05 +01:00
|
|
|
func (c clientMock) UpdateGatewayStatus(gateway *gatev1.Gateway, gatewayStatus gatev1.GatewayStatus) error {
|
2020-12-15 16:40:05 +01:00
|
|
|
for _, g := range c.gateways {
|
|
|
|
if g.Name == gateway.Name {
|
|
|
|
if !statusEquals(g.Status, gatewayStatus) {
|
|
|
|
g.Status = gatewayStatus
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return fmt.Errorf("cannot update gateway %v", gateway.Name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-01-09 10:28:05 +01:00
|
|
|
func (c clientMock) UpdateGatewayClassStatus(gatewayClass *gatev1.GatewayClass, condition metav1.Condition) error {
|
2020-12-15 16:40:05 +01:00
|
|
|
for _, gc := range c.gatewayClasses {
|
|
|
|
if gc.Name == gatewayClass.Name {
|
|
|
|
for _, c := range gc.Status.Conditions {
|
|
|
|
if c.Type == condition.Type && c.Status != condition.Status {
|
|
|
|
c.Status = condition.Status
|
|
|
|
c.LastTransitionTime = condition.LastTransitionTime
|
|
|
|
c.Message = condition.Message
|
|
|
|
c.Reason = condition.Reason
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-01-09 10:28:05 +01:00
|
|
|
func (c clientMock) UpdateGatewayStatusConditions(gateway *gatev1.Gateway, condition metav1.Condition) error {
|
2020-12-15 16:40:05 +01:00
|
|
|
for _, g := range c.gatewayClasses {
|
|
|
|
if g.Name == gateway.Name {
|
|
|
|
for _, c := range g.Status.Conditions {
|
|
|
|
if c.Type == condition.Type && (c.Status != condition.Status || c.Reason != condition.Reason) {
|
|
|
|
c.Status = condition.Status
|
|
|
|
c.LastTransitionTime = condition.LastTransitionTime
|
|
|
|
c.Message = condition.Message
|
|
|
|
c.Reason = condition.Reason
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-01-09 10:28:05 +01:00
|
|
|
func (c clientMock) GetGatewayClasses() ([]*gatev1.GatewayClass, error) {
|
2020-12-15 16:40:05 +01:00
|
|
|
return c.gatewayClasses, nil
|
|
|
|
}
|
|
|
|
|
2024-01-09 10:28:05 +01:00
|
|
|
func (c clientMock) GetGateways() []*gatev1.Gateway {
|
2020-12-15 16:40:05 +01:00
|
|
|
return c.gateways
|
|
|
|
}
|
|
|
|
|
2021-10-04 15:46:08 +02:00
|
|
|
func inNamespace(m metav1.ObjectMeta, s string) bool {
|
|
|
|
return s == metav1.NamespaceAll || m.Namespace == s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c clientMock) GetNamespaces(selector labels.Selector) ([]string, error) {
|
|
|
|
var ns []string
|
|
|
|
for _, namespace := range c.namespaces {
|
|
|
|
if selector.Matches(labels.Set(namespace.Labels)) {
|
|
|
|
ns = append(ns, namespace.Name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ns, nil
|
|
|
|
}
|
2020-12-15 16:40:05 +01:00
|
|
|
|
2024-01-09 10:28:05 +01:00
|
|
|
func (c clientMock) GetHTTPRoutes(namespaces []string) ([]*gatev1.HTTPRoute, error) {
|
|
|
|
var httpRoutes []*gatev1.HTTPRoute
|
2021-10-04 15:46:08 +02:00
|
|
|
for _, namespace := range namespaces {
|
|
|
|
for _, httpRoute := range c.httpRoutes {
|
2021-11-09 11:34:06 +01:00
|
|
|
if inNamespace(httpRoute.ObjectMeta, namespace) {
|
2021-10-04 15:46:08 +02:00
|
|
|
httpRoutes = append(httpRoutes, httpRoute)
|
|
|
|
}
|
2020-12-15 16:40:05 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return httpRoutes, nil
|
|
|
|
}
|
|
|
|
|
2023-04-17 10:56:36 +02:00
|
|
|
func (c clientMock) GetTCPRoutes(namespaces []string) ([]*gatev1alpha2.TCPRoute, error) {
|
|
|
|
var tcpRoutes []*gatev1alpha2.TCPRoute
|
2021-10-04 15:46:08 +02:00
|
|
|
for _, namespace := range namespaces {
|
|
|
|
for _, tcpRoute := range c.tcpRoutes {
|
2021-11-09 11:34:06 +01:00
|
|
|
if inNamespace(tcpRoute.ObjectMeta, namespace) {
|
2021-10-04 15:46:08 +02:00
|
|
|
tcpRoutes = append(tcpRoutes, tcpRoute)
|
|
|
|
}
|
2021-05-20 11:50:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return tcpRoutes, nil
|
|
|
|
}
|
|
|
|
|
2023-04-17 10:56:36 +02:00
|
|
|
func (c clientMock) GetTLSRoutes(namespaces []string) ([]*gatev1alpha2.TLSRoute, error) {
|
|
|
|
var tlsRoutes []*gatev1alpha2.TLSRoute
|
2021-10-04 15:46:08 +02:00
|
|
|
for _, namespace := range namespaces {
|
|
|
|
for _, tlsRoute := range c.tlsRoutes {
|
2021-11-09 11:34:06 +01:00
|
|
|
if inNamespace(tlsRoute.ObjectMeta, namespace) {
|
2021-10-04 15:46:08 +02:00
|
|
|
tlsRoutes = append(tlsRoutes, tlsRoute)
|
|
|
|
}
|
2021-05-20 11:50:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return tlsRoutes, nil
|
|
|
|
}
|
|
|
|
|
2024-01-30 16:44:05 +01:00
|
|
|
func (c clientMock) GetReferenceGrants(namespace string) ([]*gatev1beta1.ReferenceGrant, error) {
|
|
|
|
var referenceGrants []*gatev1beta1.ReferenceGrant
|
|
|
|
for _, referenceGrant := range c.referenceGrants {
|
|
|
|
if inNamespace(referenceGrant.ObjectMeta, namespace) {
|
|
|
|
referenceGrants = append(referenceGrants, referenceGrant)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return referenceGrants, nil
|
|
|
|
}
|
|
|
|
|
2020-12-15 16:40:05 +01:00
|
|
|
func (c clientMock) GetService(namespace, name string) (*corev1.Service, bool, error) {
|
|
|
|
if c.apiServiceError != nil {
|
|
|
|
return nil, false, c.apiServiceError
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, service := range c.services {
|
2021-10-04 15:46:08 +02:00
|
|
|
if inNamespace(service.ObjectMeta, namespace) && service.Name == name {
|
2020-12-15 16:40:05 +01:00
|
|
|
return service, true, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, false, c.apiServiceError
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c clientMock) GetEndpoints(namespace, name string) (*corev1.Endpoints, bool, error) {
|
|
|
|
if c.apiEndpointsError != nil {
|
|
|
|
return nil, false, c.apiEndpointsError
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, endpoints := range c.endpoints {
|
2021-10-04 15:46:08 +02:00
|
|
|
if inNamespace(endpoints.ObjectMeta, namespace) && endpoints.Name == name {
|
2020-12-15 16:40:05 +01:00
|
|
|
return endpoints, true, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return &corev1.Endpoints{}, false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c clientMock) GetSecret(namespace, name string) (*corev1.Secret, bool, error) {
|
|
|
|
if c.apiSecretError != nil {
|
|
|
|
return nil, false, c.apiSecretError
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, secret := range c.secrets {
|
2021-10-04 15:46:08 +02:00
|
|
|
if inNamespace(secret.ObjectMeta, namespace) && secret.Name == name {
|
2020-12-15 16:40:05 +01:00
|
|
|
return secret, true, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c clientMock) WatchAll(namespaces []string, stopCh <-chan struct{}) (<-chan interface{}, error) {
|
|
|
|
return c.watchChan, nil
|
|
|
|
}
|