Compare commits

...

14 commits

Author SHA1 Message Date
fd015c5a9b
Merge branch 'master' of github.com:traefik/traefik 2024-06-25 06:38:43 +05:30
27ab7b5c2e
Merge branch 'v3.0' of github.com:traefik/traefik 2024-06-25 06:38:29 +05:30
Romain
6f1bd54d86
Fix some documentation links
Co-authored-by: Kevin Pollet <pollet.kevin@gmail.com>
2024-06-24 11:22:03 +02:00
Romain
983940ae60
KubernetesGateway provider out of experimental
Co-authored-by: Kevin Pollet <pollet.kevin@gmail.com>
2024-06-24 10:36:03 +02:00
Kevin Pollet
6d8407893d
Bump Gateway API to v1.1.0
Co-authored-by: Romain <rtribotte@users.noreply.github.com>
2024-06-22 05:46:03 +02:00
Jesper Noordsij
a8a92eb2a5
Migrate to EndpointSlices API 2024-06-21 14:56:03 +02:00
mmatur
61defcdd66
Merge current v3.0 into master 2024-06-21 09:15:28 +02:00
mmatur
ec638a741e
Merge current v2.11 into v3.0 2024-06-21 08:55:31 +02:00
Michael
097e71ad24
fix: readme badge 2024-06-21 08:54:03 +02:00
Emile Vauge
eabcb3e1c0
Update maintainers 2024-06-19 17:18:03 +02:00
Kevin Pollet
53a8bd76f2
Prepare release v3.0.3 2024-06-18 16:10:06 +02:00
kevinpollet
0e89c48e38
Merge branch v2.11 into v3.0 2024-06-18 14:05:42 +02:00
Romain
385ff5055c
Prepare release v2.11.5 2024-06-18 12:00:04 +02:00
Kevin Pollet
b4f99ae3ac
Support HTTPRoute method and query param matching 2024-06-18 09:48:04 +02:00
111 changed files with 20953 additions and 13490 deletions
CHANGELOG.mdREADME.md
docs
go.modgo.sum
integration
pkg
cli
config/static
provider/kubernetes
crd
gateway
ingress
client.goclient_mock_test.goclient_test.go
fixtures
2-ingresses-in-different-namespace-with-same-service-name.ymlDouble-Single-Service-Ingress.ymlIngress-Two-rules-with-one-host-and-one-path.ymlIngress-one-rule-with-one-host-and-two-paths.ymlIngress-one-rule-with-one-path-and-one-host.ymlIngress-one-rule-with-two-paths.ymlIngress-with-IPv6-endpoints.ymlIngress-with-a-basic-rule-on-one-path-with-https-(port-==-443).ymlIngress-with-a-basic-rule-on-one-path-with-https-(portname-==-https).ymlIngress-with-a-basic-rule-on-one-path-with-https-(portname-starts-with-https).ymlIngress-with-a-basic-rule-on-one-path.ymlIngress-with-a-named-port-matching-subset-of-service-pods.ymlIngress-with-annotations.ymlIngress-with-conflicting-routers-on-host.ymlIngress-with-conflicting-routers-on-path.ymlIngress-with-default-traefik-ingressClass.ymlIngress-with-defaultbackend.ymlIngress-with-empty-pathType.ymlIngress-with-exact-pathType.ymlIngress-with-implementationSpecific-pathType.ymlIngress-with-ingress-annotation.ymlIngress-with-ingressClass-without-annotation.ymlIngress-with-ingressClass.ymlIngress-with-ingressClasses-filter.ymlIngress-with-missing-ingressClass.ymlIngress-with-multiple-ingressClasses.ymlIngress-with-named-port.ymlIngress-with-non-matching-provider-traefik-ingressClass-and-annotation.ymlIngress-with-one-host-without-path.ymlIngress-with-one-service-without-endpoints-subset.ymlIngress-with-port-invalid-for-one-service.ymlIngress-with-port-name-in-backend-and-2-pod-replica.ymlIngress-with-port-name-in-backend-and-no-pod-replica.ymlIngress-with-port-value-in-backend-and-no-pod-replica.ymlIngress-with-prefix-pathType.ymlIngress-with-two-different-rules-with-one-path.ymlIngress-with-two-paths-using-same-service-and-different-port-name.ymlIngress-with-two-services.yml

View file

@ -1,3 +1,19 @@
## [v3.0.3](https://github.com/traefik/traefik/tree/v3.0.3) (2024-06-18)
[All Commits](https://github.com/traefik/traefik/compare/v3.0.2...v3.0.3)
**Misc:**
- Merge v2.11 into v3.0 ([#10823](https://github.com/traefik/traefik/pull/10823) by [kevinpollet](https://github.com/kevinpollet))
- Merge v2.11 into v3.0 ([#10810](https://github.com/traefik/traefik/pull/10810) by [mmatur](https://github.com/mmatur))
## [v2.11.5](https://github.com/traefik/traefik/tree/v2.11.5) (2024-06-18)
[All Commits](https://github.com/traefik/traefik/compare/v2.11.4...v2.11.5)
**Bug fixes:**
- **[acme]** Update go-acme/lego to v4.17.4 ([#10803](https://github.com/traefik/traefik/pull/10803) by [ldez](https://github.com/ldez))
**Documentation:**
- Update the supported versions table ([#10798](https://github.com/traefik/traefik/pull/10798) by [nmengin](https://github.com/nmengin))
## [v3.0.2](https://github.com/traefik/traefik/tree/v3.0.2) (2024-06-10) ## [v3.0.2](https://github.com/traefik/traefik/tree/v3.0.2) (2024-06-10)
[All Commits](https://github.com/traefik/traefik/compare/v3.0.1...v3.0.2) [All Commits](https://github.com/traefik/traefik/compare/v3.0.1...v3.0.2)

View file

@ -7,7 +7,7 @@
</picture> </picture>
</p> </p>
[![Build Status SemaphoreCI](https://semaphoreci.com/api/v1/containous/traefik/branches/master/shields_badge.svg)](https://semaphoreci.com/containous/traefik) [![Build Status SemaphoreCI](https://traefik-oss.semaphoreci.com/badges/traefik/branches/master.svg?style=shields)](https://traefik-oss.semaphoreci.com/projects/traefik)
[![Docs](https://img.shields.io/badge/docs-current-brightgreen.svg)](https://doc.traefik.io/traefik) [![Docs](https://img.shields.io/badge/docs-current-brightgreen.svg)](https://doc.traefik.io/traefik)
[![Go Report Card](https://goreportcard.com/badge/traefik/traefik)](https://goreportcard.com/report/traefik/traefik) [![Go Report Card](https://goreportcard.com/badge/traefik/traefik)](https://goreportcard.com/report/traefik/traefik)
[![License](https://img.shields.io/badge/license-MIT-blue.svg)](https://github.com/traefik/traefik/blob/master/LICENSE.md) [![License](https://img.shields.io/badge/license-MIT-blue.svg)](https://github.com/traefik/traefik/blob/master/LICENSE.md)

View file

@ -15,7 +15,7 @@ Let's see how.
### General ### General
This [documentation](https://doc.traefik.io/traefik/ "Link to the official Traefik documentation") is built with [MkDocs](https://mkdocs.org/ "Link to website of MkDocs"). This [documentation](../../ "Link to the official Traefik documentation") is built with [MkDocs](https://mkdocs.org/ "Link to website of MkDocs").
### Method 1: `Docker` and `make` ### Method 1: `Docker` and `make`

View file

@ -9,7 +9,6 @@ description: "Traefik Proxy is an open source software with a thriving community
* Emile Vauge [@emilevauge](https://github.com/emilevauge) * Emile Vauge [@emilevauge](https://github.com/emilevauge)
* Manuel Zapf [@SantoDE](https://github.com/SantoDE) * Manuel Zapf [@SantoDE](https://github.com/SantoDE)
* Ludovic Fernandez [@ldez](https://github.com/ldez)
* Julien Salleyron [@juliens](https://github.com/juliens) * Julien Salleyron [@juliens](https://github.com/juliens)
* Nicolas Mengin [@nmengin](https://github.com/nmengin) * Nicolas Mengin [@nmengin](https://github.com/nmengin)
* Michaël Matur [@mmatur](https://github.com/mmatur) * Michaël Matur [@mmatur](https://github.com/mmatur)
@ -33,6 +32,7 @@ People who have had an incredibly positive impact on the project, and are now fo
* Daniel Tomcej [@dtomcej](https://github.com/dtomcej) * Daniel Tomcej [@dtomcej](https://github.com/dtomcej)
* Timo Reimann [@timoreimann](https://github.com/timoreimann) * Timo Reimann [@timoreimann](https://github.com/timoreimann)
* Marco Jantke [@mjantke](https://github.com/mjeri) * Marco Jantke [@mjantke](https://github.com/mjeri)
* Ludovic Fernandez [@ldez](https://github.com/ldez)
## Maintainer's Guidelines ## Maintainer's Guidelines

View file

@ -35,12 +35,18 @@ rules:
- "" - ""
resources: resources:
- services - services
- endpoints
- secrets - secrets
verbs: verbs:
- get - get
- list - list
- watch - watch
- apiGroups:
- discovery.k8s.io
resources:
- endpointslices
verbs:
- list
- watch
- apiGroups: - apiGroups:
- extensions - extensions
- networking.k8s.io - networking.k8s.io

View file

@ -135,7 +135,7 @@ It is now unsupported and would prevent Traefik to start.
##### Remediation ##### Remediation
The `http3` option should be removed from the static configuration experimental section. The `http3` option should be removed from the static configuration experimental section.
To configure `http3`, please checkout the [entrypoint configuration documentation](https://doc.traefik.io/traefik/v3.0/routing/entrypoints/#http3_1). To configure `http3`, please checkout the [entrypoint configuration documentation](../routing/entrypoints.md#http3_1).
### Consul provider ### Consul provider

View file

@ -29,7 +29,7 @@ core:
defaultRuleSyntax: v2 defaultRuleSyntax: v2
``` ```
This snippet in the static configuration makes the [v2 format](https://doc.traefik.io/traefik/v3.0/migration/v2-to-v3/?ref=traefik.io#configure-the-default-syntax-in-static-configuration "Link to configure default syntax in static config") the default rule matchers syntax. This snippet in the static configuration makes the [v2 format](../migration/v2-to-v3-details.md#configure-the-default-syntax-in-static-configuration "Link to configure default syntax in static config") the default rule matchers syntax.
Start Traefik v3 with this new configuration to test it. Start Traefik v3 with this new configuration to test it.

View file

@ -553,7 +553,7 @@ The following ciphers have been removed from the default list:
- `TLS_RSA_WITH_AES_128_GCM_SHA256` - `TLS_RSA_WITH_AES_128_GCM_SHA256`
- `TLS_RSA_WITH_AES_256_GCM_SHA384` - `TLS_RSA_WITH_AES_256_GCM_SHA384`
To enable these ciphers, please set the option `CipherSuites` in your [TLS configuration](https://doc.traefik.io/traefik/https/tls/#cipher-suites) or set the environment variable `GODEBUG=tlsrsakex=1`. To enable these ciphers, please set the option `CipherSuites` in your [TLS configuration](../https/tls.md#cipher-suites) or set the environment variable `GODEBUG=tlsrsakex=1`.
### Minimum TLS Version ### Minimum TLS Version
@ -562,7 +562,7 @@ To enable these ciphers, please set the option `CipherSuites` in your [TLS confi
> This change can be reverted with the `tls10server=1 GODEBUG` setting. > This change can be reverted with the `tls10server=1 GODEBUG` setting.
> (https://go.dev/doc/go1.22#crypto/tls) > (https://go.dev/doc/go1.22#crypto/tls)
To enable TLS 1.0, please set the option `MinVersion` to `VersionTLS10` in your [TLS configuration](https://doc.traefik.io/traefik/https/tls/#cipher-suites) or set the environment variable `GODEBUG=tls10server=1`. To enable TLS 1.0, please set the option `MinVersion` to `VersionTLS10` in your [TLS configuration](../https/tls.md#cipher-suites) or set the environment variable `GODEBUG=tls10server=1`.
## v2.11.1 ## v2.11.1

View file

@ -0,0 +1,53 @@
---
title: "Traefik Migration Documentation"
description: "Learn the steps needed to migrate to new Traefik Proxy v3 versions. Read the technical documentation."
---
# Migration: Steps needed between the versions
## v3.0 to v3.1
### Kubernetes Provider RBACs
Starting with v3.1, the Kubernetes Providers now use the [EndpointSlices API](https://kubernetes.io/docs/concepts/services-networking/endpoint-slices/) (Kubernetes >=v1.21) to discover service endpoint addresses.
Therefore, in the corresponding RBACs (see [KubernetesIngress](../routing/providers/kubernetes-ingress.md#configuration-example), [KubernetesCRD](../reference/dynamic-configuration/kubernetes-crd.md#rbac), and [KubernetesGateway](../reference/dynamic-configuration/kubernetes-gateway.md#rbac) provider RBACs),
the `endpoints` right has to be removed and the following `endpointslices` right has to be added.
```yaml
...
- apiGroups:
- discovery.k8s.io
resources:
- endpointslices
verbs:
- list
- watch
...
```
#### Gateway API: KubernetesGateway Provider
In v3.1, the KubernetesGateway Provider is no longer an experimental feature.
It can be enabled without the associated `experimental.kubernetesgateway` option, which is now deprecated.
??? example "An example of the experimental `kubernetesgateway` option"
```yaml tab="File (YAML)"
experimental:
kubernetesgateway: true
```
```toml tab="File (TOML)"
[experimental]
kubernetesgateway=true
```
```bash tab="CLI"
--experimental.kubernetesgateway=true
```
##### Remediation
The `kubernetesgateway` option should be removed from the experimental section of the static configuration.
To configure `kubernetesgateway`, please check out the [KubernetesGateway Provider documentation](../providers/kubernetes-gateway.md).

View file

@ -58,7 +58,7 @@ For this reason, users can run multiple instances of Traefik at the same time to
When using a single instance of Traefik with Let's Encrypt, you should encounter no issues. However, this could be a single point of failure. When using a single instance of Traefik with Let's Encrypt, you should encounter no issues. However, this could be a single point of failure.
Unfortunately, it is not possible to run multiple instances of Traefik Proxy 2.0 with Let's Encrypt enabled, because there is no way to ensure that the correct instance of Traefik will receive the challenge request and subsequent responses. Unfortunately, it is not possible to run multiple instances of Traefik Proxy 2.0 with Let's Encrypt enabled, because there is no way to ensure that the correct instance of Traefik will receive the challenge request and subsequent responses.
Previous versions of Traefik used a [KV store](https://doc.traefik.io/traefik/v1.7/configuration/acme/#storage) to attempt to achieve this, but due to sub-optimal performance that feature was dropped in 2.0. Early versions (v1.x) of Traefik used a [KV store](https://doc.traefik.io/traefik/v1.7/configuration/acme/#storage) to attempt to achieve this, but due to sub-optimal performance that feature was dropped in 2.0.
If you need Let's Encrypt with HA in a Kubernetes environment, we recommend using [Traefik Enterprise](https://traefik.io/traefik-enterprise/), which includes distributed Let's Encrypt as a supported feature. If you need Let's Encrypt with HA in a Kubernetes environment, we recommend using [Traefik Enterprise](https://traefik.io/traefik-enterprise/), which includes distributed Let's Encrypt as a supported feature.
@ -183,7 +183,7 @@ _Optional, Default: ""_
A label selector can be defined to filter on specific resource objects only, A label selector can be defined to filter on specific resource objects only,
this applies only to Traefik [Custom Resources](../routing/providers/kubernetes-crd.md#custom-resource-definition-crd) this applies only to Traefik [Custom Resources](../routing/providers/kubernetes-crd.md#custom-resource-definition-crd)
and has no effect on Kubernetes `Secrets`, `Endpoints` and `Services`. and has no effect on Kubernetes `Secrets`, `EndpointSlices` and `Services`.
If left empty, Traefik processes all resource objects in the configured namespaces. If left empty, Traefik processes all resource objects in the configured namespaces.
See [label-selectors](https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors) for details. See [label-selectors](https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors) for details.

View file

@ -80,7 +80,7 @@ When using a single instance of Traefik Proxy with Let's Encrypt, you should enc
However, this could be a single point of failure. However, this could be a single point of failure.
Unfortunately, it is not possible to run multiple instances of Traefik 2.0 with Let's Encrypt enabled, Unfortunately, it is not possible to run multiple instances of Traefik 2.0 with Let's Encrypt enabled,
because there is no way to ensure that the correct instance of Traefik receives the challenge request, and subsequent responses. because there is no way to ensure that the correct instance of Traefik receives the challenge request, and subsequent responses.
Previous versions of Traefik used a [KV store](https://doc.traefik.io/traefik/v1.7/configuration/acme/#storage) to attempt to achieve this, Early versions (v1.x) of Traefik used a [KV store](https://doc.traefik.io/traefik/v1.7/configuration/acme/#storage) to attempt to achieve this,
but due to sub-optimal performance that feature was dropped in 2.0. but due to sub-optimal performance that feature was dropped in 2.0.
If you need Let's Encrypt with high availability in a Kubernetes environment, If you need Let's Encrypt with high availability in a Kubernetes environment,

View file

@ -150,8 +150,8 @@ Below is the list of the currently supported providers in Traefik.
!!! info "More Providers" !!! info "More Providers"
The current version of Traefik does not yet support every provider that Traefik v1.7 did. The current version of Traefik does not yet support every provider that Traefik v2.11 did.
See the [previous version (v1.7)](https://doc.traefik.io/traefik/v1.7/) for more providers. See the [previous version (v2.11)](https://doc.traefik.io/traefik/v2.11/) for more information.
### Configuration Reload Frequency ### Configuration Reload Frequency

View file

@ -8,13 +8,19 @@ rules:
- "" - ""
resources: resources:
- services - services
- endpoints
- secrets - secrets
- nodes - nodes
verbs: verbs:
- get - get
- list - list
- watch - watch
- apiGroups:
- discovery.k8s.io
resources:
- endpointslices
verbs:
- list
- watch
- apiGroups: - apiGroups:
- extensions - extensions
- networking.k8s.io - networking.k8s.io

View file

@ -15,12 +15,18 @@ rules:
- "" - ""
resources: resources:
- services - services
- endpoints
- secrets - secrets
verbs: verbs:
- get - get
- list - list
- watch - watch
- apiGroups:
- discovery.k8s.io
resources:
- endpointslices
verbs:
- list
- watch
- apiGroups: - apiGroups:
- gateway.networking.k8s.io - gateway.networking.k8s.io
resources: resources:

View file

@ -211,7 +211,7 @@ WriteTimeout is the maximum duration before timing out writes of the response. I
Timeout defines how long to wait on an idle session before releasing the related resources. (Default: ```3```) Timeout defines how long to wait on an idle session before releasing the related resources. (Default: ```3```)
`--experimental.kubernetesgateway`: `--experimental.kubernetesgateway`:
Allow the Kubernetes gateway api provider usage. (Default: ```false```) (Deprecated) Allow the Kubernetes gateway api provider usage. (Default: ```false```)
`--experimental.localplugins.<name>`: `--experimental.localplugins.<name>`:
Local plugins configuration. (Default: ```false```) Local plugins configuration. (Default: ```false```)

View file

@ -211,7 +211,7 @@ WriteTimeout is the maximum duration before timing out writes of the response. I
Timeout defines how long to wait on an idle session before releasing the related resources. (Default: ```3```) Timeout defines how long to wait on an idle session before releasing the related resources. (Default: ```3```)
`TRAEFIK_EXPERIMENTAL_KUBERNETESGATEWAY`: `TRAEFIK_EXPERIMENTAL_KUBERNETESGATEWAY`:
Allow the Kubernetes gateway api provider usage. (Default: ```false```) (Deprecated) Allow the Kubernetes gateway api provider usage. (Default: ```false```)
`TRAEFIK_EXPERIMENTAL_LOCALPLUGINS_<NAME>`: `TRAEFIK_EXPERIMENTAL_LOCALPLUGINS_<NAME>`:
Local plugins configuration. (Default: ```false```) Local plugins configuration. (Default: ```false```)

View file

@ -29,12 +29,18 @@ which in turn will create the resulting routers, services, handlers, etc.
- "" - ""
resources: resources:
- services - services
- endpoints
- secrets - secrets
verbs: verbs:
- get - get
- list - list
- watch - watch
- apiGroups:
- discovery.k8s.io
resources:
- endpointslices
verbs:
- list
- watch
- apiGroups: - apiGroups:
- extensions - extensions
- networking.k8s.io - networking.k8s.io
@ -427,12 +433,19 @@ This way, any Ingress attached to this Entrypoint will have TLS termination by d
- "" - ""
resources: resources:
- services - services
- endpoints
- secrets - secrets
verbs: verbs:
- get - get
- list - list
- watch - watch
- apiGroups:
- discovery.k8s.io
resources:
- endpointslices
verbs:
- get
- list
- watch
- apiGroups: - apiGroups:
- extensions - extensions
- networking.k8s.io - networking.k8s.io
@ -612,12 +625,19 @@ For more options, please refer to the available [annotations](#on-ingress).
- "" - ""
resources: resources:
- services - services
- endpoints
- secrets - secrets
verbs: verbs:
- get - get
- list - list
- watch - watch
- apiGroups:
- discovery.k8s.io
resources:
- endpointslices
verbs:
- get
- list
- watch
- apiGroups: - apiGroups:
- extensions - extensions
- networking.k8s.io - networking.k8s.io
@ -832,7 +852,7 @@ TLS certificates can be managed in Secrets objects.
whether the LB's children are directly the pods IPs or if the only child is the Kubernetes Service clusterIP. whether the LB's children are directly the pods IPs or if the only child is the Kubernetes Service clusterIP.
One alternative is to use an `ExternalName` service to forward requests to the Kubernetes service through DNS. One alternative is to use an `ExternalName` service to forward requests to the Kubernetes service through DNS.
To do so, one must [allow external name services](https://doc.traefik.io/traefik/providers/kubernetes-ingress/#allowexternalnameservices "Link to docs about allowing external name services"). To do so, one must [allow external name services](../providers/kubernetes-ingress/#allowexternalnameservices "Link to docs about allowing external name services").
Traefik automatically requests endpoint information based on the service provided in the ingress spec. Traefik automatically requests endpoint information based on the service provided in the ingress spec.
Although Traefik will connect directly to the endpoints (pods), Although Traefik will connect directly to the endpoints (pods),

View file

@ -172,6 +172,7 @@ nav:
- 'HTTP Challenge': 'user-guides/docker-compose/acme-http/index.md' - 'HTTP Challenge': 'user-guides/docker-compose/acme-http/index.md'
- 'DNS Challenge': 'user-guides/docker-compose/acme-dns/index.md' - 'DNS Challenge': 'user-guides/docker-compose/acme-dns/index.md'
- 'Migration': - 'Migration':
- 'Traefik v3 minor migrations': 'migration/v3.md'
- 'Traefik v2 to v3': - 'Traefik v2 to v3':
- 'Migration guide': 'migration/v2-to-v3.md' - 'Migration guide': 'migration/v2-to-v3.md'
- 'Configuration changes for v3': 'migration/v2-to-v3-details.md' - 'Configuration changes for v3': 'migration/v2-to-v3-details.md'

42
go.mod
View file

@ -1,6 +1,6 @@
module github.com/traefik/traefik/v3 module github.com/traefik/traefik/v3
go 1.22 go 1.22.4
require ( require (
github.com/BurntSushi/toml v1.4.0 github.com/BurntSushi/toml v1.4.0
@ -22,7 +22,7 @@ require (
github.com/golang/protobuf v1.5.4 github.com/golang/protobuf v1.5.4
github.com/google/go-github/v28 v28.1.1 github.com/google/go-github/v28 v28.1.1
github.com/gorilla/mux v1.8.0 github.com/gorilla/mux v1.8.0
github.com/gorilla/websocket v1.5.0 github.com/gorilla/websocket v1.5.1
github.com/hashicorp/consul/api v1.26.1 github.com/hashicorp/consul/api v1.26.1
github.com/hashicorp/go-hclog v1.6.3 github.com/hashicorp/go-hclog v1.6.3
github.com/hashicorp/go-multierror v1.1.1 github.com/hashicorp/go-multierror v1.1.1
@ -79,7 +79,7 @@ require (
go.opentelemetry.io/otel/sdk v1.27.0 go.opentelemetry.io/otel/sdk v1.27.0
go.opentelemetry.io/otel/sdk/metric v1.27.0 go.opentelemetry.io/otel/sdk/metric v1.27.0
go.opentelemetry.io/otel/trace v1.27.0 go.opentelemetry.io/otel/trace v1.27.0
golang.org/x/exp v0.0.0-20240404231335-c0f41cb1a7a0 golang.org/x/exp v0.0.0-20240416160154-fe59bbe5cc7f
golang.org/x/mod v0.18.0 golang.org/x/mod v0.18.0
golang.org/x/net v0.26.0 golang.org/x/net v0.26.0
golang.org/x/sys v0.21.0 golang.org/x/sys v0.21.0
@ -88,14 +88,14 @@ require (
golang.org/x/tools v0.22.0 golang.org/x/tools v0.22.0
google.golang.org/grpc v1.64.0 google.golang.org/grpc v1.64.0
gopkg.in/yaml.v3 v3.0.1 gopkg.in/yaml.v3 v3.0.1
k8s.io/api v0.29.2 k8s.io/api v0.30.0
k8s.io/apiextensions-apiserver v0.28.3 k8s.io/apiextensions-apiserver v0.30.0
k8s.io/apimachinery v0.29.2 k8s.io/apimachinery v0.30.0
k8s.io/client-go v0.29.2 k8s.io/client-go v0.30.0
k8s.io/utils v0.0.0-20230726121419-3b25d923346b k8s.io/utils v0.0.0-20240423183400-0849a56e8f22
mvdan.cc/xurls/v2 v2.5.0 mvdan.cc/xurls/v2 v2.5.0
sigs.k8s.io/controller-runtime v0.16.3 sigs.k8s.io/controller-runtime v0.18.0
sigs.k8s.io/gateway-api v1.0.0 sigs.k8s.io/gateway-api v1.1.0
) )
require ( require (
@ -165,9 +165,9 @@ require (
github.com/distribution/reference v0.5.0 // indirect github.com/distribution/reference v0.5.0 // indirect
github.com/dnsimple/dnsimple-go v1.7.0 // indirect github.com/dnsimple/dnsimple-go v1.7.0 // indirect
github.com/docker/go-units v0.5.0 // indirect github.com/docker/go-units v0.5.0 // indirect
github.com/emicklei/go-restful/v3 v3.11.0 // indirect github.com/emicklei/go-restful/v3 v3.12.0 // indirect
github.com/evanphx/json-patch v5.7.0+incompatible // indirect github.com/evanphx/json-patch v5.7.0+incompatible // indirect
github.com/evanphx/json-patch/v5 v5.7.0 // indirect github.com/evanphx/json-patch/v5 v5.9.0 // indirect
github.com/exoscale/egoscale v0.102.3 // indirect github.com/exoscale/egoscale v0.102.3 // indirect
github.com/fatih/color v1.16.0 // indirect github.com/fatih/color v1.16.0 // indirect
github.com/felixge/httpsnoop v1.0.4 // indirect github.com/felixge/httpsnoop v1.0.4 // indirect
@ -178,11 +178,11 @@ require (
github.com/go-logfmt/logfmt v0.5.1 // indirect github.com/go-logfmt/logfmt v0.5.1 // indirect
github.com/go-logr/logr v1.4.1 // indirect github.com/go-logr/logr v1.4.1 // indirect
github.com/go-logr/stdr v1.2.2 // indirect github.com/go-logr/stdr v1.2.2 // indirect
github.com/go-logr/zapr v1.2.4 // indirect github.com/go-logr/zapr v1.3.0 // indirect
github.com/go-ole/go-ole v1.2.6 // indirect github.com/go-ole/go-ole v1.2.6 // indirect
github.com/go-openapi/jsonpointer v0.20.0 // indirect github.com/go-openapi/jsonpointer v0.21.0 // indirect
github.com/go-openapi/jsonreference v0.20.2 // indirect github.com/go-openapi/jsonreference v0.21.0 // indirect
github.com/go-openapi/swag v0.22.4 // indirect github.com/go-openapi/swag v0.23.0 // indirect
github.com/go-resty/resty/v2 v2.11.0 // indirect github.com/go-resty/resty/v2 v2.11.0 // indirect
github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 // indirect github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 // indirect
github.com/go-viper/mapstructure/v2 v2.0.0 // indirect github.com/go-viper/mapstructure/v2 v2.0.0 // indirect
@ -316,9 +316,9 @@ require (
github.com/yandex-cloud/go-sdk v0.0.0-20240318084659-dfa50323a0b4 // indirect github.com/yandex-cloud/go-sdk v0.0.0-20240318084659-dfa50323a0b4 // indirect
github.com/yusufpapurcu/wmi v1.2.3 // indirect github.com/yusufpapurcu/wmi v1.2.3 // indirect
github.com/zeebo/errs v1.2.2 // indirect github.com/zeebo/errs v1.2.2 // indirect
go.etcd.io/etcd/api/v3 v3.5.9 // indirect go.etcd.io/etcd/api/v3 v3.5.10 // indirect
go.etcd.io/etcd/client/pkg/v3 v3.5.9 // indirect go.etcd.io/etcd/client/pkg/v3 v3.5.10 // indirect
go.etcd.io/etcd/client/v3 v3.5.9 // indirect go.etcd.io/etcd/client/v3 v3.5.10 // indirect
go.opencensus.io v0.24.0 // indirect go.opencensus.io v0.24.0 // indirect
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0 // indirect go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0 // indirect
go.opentelemetry.io/contrib/propagators/aws v1.27.0 // indirect go.opentelemetry.io/contrib/propagators/aws v1.27.0 // indirect
@ -347,8 +347,8 @@ require (
gopkg.in/ns1/ns1-go.v2 v2.9.1 // indirect gopkg.in/ns1/ns1-go.v2 v2.9.1 // indirect
gopkg.in/square/go-jose.v2 v2.5.1 // indirect gopkg.in/square/go-jose.v2 v2.5.1 // indirect
gopkg.in/yaml.v2 v2.4.0 // indirect gopkg.in/yaml.v2 v2.4.0 // indirect
k8s.io/klog/v2 v2.110.1 // indirect k8s.io/klog/v2 v2.120.1 // indirect
k8s.io/kube-openapi v0.0.0-20231010175941-2dd684a91f00 // indirect k8s.io/kube-openapi v0.0.0-20240423202451-8948a665c108 // indirect
nhooyr.io/websocket v1.8.7 // indirect nhooyr.io/websocket v1.8.7 // indirect
sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd // indirect sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd // indirect
sigs.k8s.io/structured-merge-diff/v4 v4.4.1 // indirect sigs.k8s.io/structured-merge-diff/v4 v4.4.1 // indirect

96
go.sum
View file

@ -151,7 +151,6 @@ github.com/aws/aws-sdk-go-v2/service/sts v1.28.12 h1:M/1u4HBpwLuMtjlxuI2y6HoVLzF
github.com/aws/aws-sdk-go-v2/service/sts v1.28.12/go.mod h1:kcfd+eTdEi/40FIbLq4Hif3XMXnl5b/+t/KTfLt9xIk= github.com/aws/aws-sdk-go-v2/service/sts v1.28.12/go.mod h1:kcfd+eTdEi/40FIbLq4Hif3XMXnl5b/+t/KTfLt9xIk=
github.com/aws/smithy-go v1.20.2 h1:tbp628ireGtzcHDDmLT/6ADHidqnwgF57XOXZe6tp4Q= github.com/aws/smithy-go v1.20.2 h1:tbp628ireGtzcHDDmLT/6ADHidqnwgF57XOXZe6tp4Q=
github.com/aws/smithy-go v1.20.2/go.mod h1:krry+ya/rV9RDcV/Q16kpu6ypI4K2czasz0NC3qS14E= github.com/aws/smithy-go v1.20.2/go.mod h1:krry+ya/rV9RDcV/Q16kpu6ypI4K2czasz0NC3qS14E=
github.com/benbjohnson/clock v1.1.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA=
github.com/benbjohnson/clock v1.3.0 h1:ip6w0uFQkncKQ979AypyG0ER7mqUSBdKLOgAle/AT8A= github.com/benbjohnson/clock v1.3.0 h1:ip6w0uFQkncKQ979AypyG0ER7mqUSBdKLOgAle/AT8A=
github.com/benbjohnson/clock v1.3.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= github.com/benbjohnson/clock v1.3.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA=
github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q=
@ -282,8 +281,8 @@ github.com/eapache/queue v1.1.0/go.mod h1:6eCeP0CKFpHLu8blIFXhExK/dRa7WDZfr6jVFP
github.com/edsrzf/mmap-go v1.0.0/go.mod h1:YO35OhQPt3KJa3ryjFM5Bs14WD66h8eGKpfaBNrHW5M= github.com/edsrzf/mmap-go v1.0.0/go.mod h1:YO35OhQPt3KJa3ryjFM5Bs14WD66h8eGKpfaBNrHW5M=
github.com/eknkc/amber v0.0.0-20171010120322-cdade1c07385 h1:clC1lXBpe2kTj2VHdaIu9ajZQe4kcEY9j0NsnDDBZ3o= github.com/eknkc/amber v0.0.0-20171010120322-cdade1c07385 h1:clC1lXBpe2kTj2VHdaIu9ajZQe4kcEY9j0NsnDDBZ3o=
github.com/eknkc/amber v0.0.0-20171010120322-cdade1c07385/go.mod h1:0vRUJqYpeSZifjYj7uP3BG/gKcuzL9xWVV/Y+cK33KM= github.com/eknkc/amber v0.0.0-20171010120322-cdade1c07385/go.mod h1:0vRUJqYpeSZifjYj7uP3BG/gKcuzL9xWVV/Y+cK33KM=
github.com/emicklei/go-restful/v3 v3.11.0 h1:rAQeMHw1c7zTmncogyy8VvRZwtkmkZ4FxERmMY4rD+g= github.com/emicklei/go-restful/v3 v3.12.0 h1:y2DdzBAURM29NFF94q6RaY4vjIH1rtwDapwQtU84iWk=
github.com/emicklei/go-restful/v3 v3.11.0/go.mod h1:6n3XBCmQQb25CM2LCACGz8ukIrRry+4bhvbpWn3mrbc= github.com/emicklei/go-restful/v3 v3.12.0/go.mod h1:6n3XBCmQQb25CM2LCACGz8ukIrRry+4bhvbpWn3mrbc=
github.com/envoyproxy/go-control-plane v0.6.9/go.mod h1:SBwIajubJHhxtWwsL9s8ss4safvEdbitLhGGK48rN6g= github.com/envoyproxy/go-control-plane v0.6.9/go.mod h1:SBwIajubJHhxtWwsL9s8ss4safvEdbitLhGGK48rN6g=
github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
@ -295,8 +294,8 @@ github.com/envoyproxy/go-control-plane v0.10.2-0.20220325020618-49ff273808a1/go.
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
github.com/evanphx/json-patch v5.7.0+incompatible h1:vgGkfT/9f8zE6tvSCe74nfpAVDQ2tG6yudJd8LBksgI= github.com/evanphx/json-patch v5.7.0+incompatible h1:vgGkfT/9f8zE6tvSCe74nfpAVDQ2tG6yudJd8LBksgI=
github.com/evanphx/json-patch v5.7.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk= github.com/evanphx/json-patch v5.7.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk=
github.com/evanphx/json-patch/v5 v5.7.0 h1:nJqP7uwL84RJInrohHfW0Fx3awjbm8qZeFv0nW9SYGc= github.com/evanphx/json-patch/v5 v5.9.0 h1:kcBlZQbplgElYIlo/n1hJbls2z/1awpXxpRi0/FOJfg=
github.com/evanphx/json-patch/v5 v5.7.0/go.mod h1:VNkHZ/282BpEyt/tObQO8s5CMPmYYq14uClGH4abBuQ= github.com/evanphx/json-patch/v5 v5.9.0/go.mod h1:VNkHZ/282BpEyt/tObQO8s5CMPmYYq14uClGH4abBuQ=
github.com/exoscale/egoscale v0.102.3 h1:DYqN2ipoLKpiFoprRGQkp2av/Ze7sUYYlGhi1N62tfY= github.com/exoscale/egoscale v0.102.3 h1:DYqN2ipoLKpiFoprRGQkp2av/Ze7sUYYlGhi1N62tfY=
github.com/exoscale/egoscale v0.102.3/go.mod h1:RPf2Gah6up+6kAEayHTQwqapzXlm93f0VQas/UEGU5c= github.com/exoscale/egoscale v0.102.3/go.mod h1:RPf2Gah6up+6kAEayHTQwqapzXlm93f0VQas/UEGU5c=
github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4=
@ -349,26 +348,22 @@ github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG
github.com/go-logfmt/logfmt v0.5.1 h1:otpy5pqBCBZ1ng9RQ0dPu4PN7ba75Y/aA+UpowDyNVA= github.com/go-logfmt/logfmt v0.5.1 h1:otpy5pqBCBZ1ng9RQ0dPu4PN7ba75Y/aA+UpowDyNVA=
github.com/go-logfmt/logfmt v0.5.1/go.mod h1:WYhtIu8zTZfxdn5+rREduYbwxfcBr/Vr6KEVveWlfTs= github.com/go-logfmt/logfmt v0.5.1/go.mod h1:WYhtIu8zTZfxdn5+rREduYbwxfcBr/Vr6KEVveWlfTs=
github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A=
github.com/go-logr/logr v1.2.4/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A=
github.com/go-logr/logr v1.3.0/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
github.com/go-logr/logr v1.4.1 h1:pKouT5E8xu9zeFC39JXRDukb6JFQPXM5p5I91188VAQ= github.com/go-logr/logr v1.4.1 h1:pKouT5E8xu9zeFC39JXRDukb6JFQPXM5p5I91188VAQ=
github.com/go-logr/logr v1.4.1/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY= github.com/go-logr/logr v1.4.1/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag= github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag=
github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE= github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE=
github.com/go-logr/zapr v1.2.4 h1:QHVo+6stLbfJmYGkQ7uGHUCu5hnAFAj6mDe6Ea0SeOo= github.com/go-logr/zapr v1.3.0 h1:XGdV8XW8zdwFiwOA2Dryh1gj2KRQyOOoNmBy4EplIcQ=
github.com/go-logr/zapr v1.2.4/go.mod h1:FyHWQIzQORZ0QVE1BtVHv3cKtNLuXsbNLtpuhNapBOA= github.com/go-logr/zapr v1.3.0/go.mod h1:YKepepNBd1u/oyhd/yQmtjVXmm9uML4IXUgMOwR8/Gg=
github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY= github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY=
github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0=
github.com/go-openapi/jsonpointer v0.19.5/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= github.com/go-openapi/jsonpointer v0.19.5/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg=
github.com/go-openapi/jsonpointer v0.19.6/go.mod h1:osyAmYz/mB/C3I+WsTTSgw1ONzaLJoLCyoi6/zppojs= github.com/go-openapi/jsonpointer v0.21.0 h1:YgdVicSA9vH5RiHs9TZW5oyafXZFc6+2Vc1rr/O9oNQ=
github.com/go-openapi/jsonpointer v0.20.0 h1:ESKJdU9ASRfaPNOPRx12IUyA1vn3R9GiE3KYD14BXdQ= github.com/go-openapi/jsonpointer v0.21.0/go.mod h1:IUyH9l/+uyhIYQ/PXVA41Rexl+kOkAPDdXEYns6fzUY=
github.com/go-openapi/jsonpointer v0.20.0/go.mod h1:6PGzBjjIIumbLYysB73Klnms1mwnU4G3YHOECG3CedA= github.com/go-openapi/jsonreference v0.21.0 h1:Rs+Y7hSXT83Jacb7kFyjn4ijOuVGSvOdF2+tg1TRrwQ=
github.com/go-openapi/jsonreference v0.20.2 h1:3sVjiK66+uXK/6oQ8xgcRKcFgQ5KXa2KvnJRumpMGbE= github.com/go-openapi/jsonreference v0.21.0/go.mod h1:LmZmgsrTkVg9LG4EaHeY8cBDslNPMo06cago5JNLkm4=
github.com/go-openapi/jsonreference v0.20.2/go.mod h1:Bl1zwGIM8/wsvqjsOQLJ/SH+En5Ap4rVB5KVcIDZG2k=
github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk= github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk=
github.com/go-openapi/swag v0.22.3/go.mod h1:UzaqsxGiab7freDnrUUra0MwWfN/q7tE4j+VcZ0yl14= github.com/go-openapi/swag v0.23.0 h1:vsEVJDUo2hPJ2tu0/Xc+4noaxyEffXNIs3cOULZ+GrE=
github.com/go-openapi/swag v0.22.4 h1:QLMzNJnMGPRNDCbySlcj1x01tzU8/9LTTL9hZZZogBU= github.com/go-openapi/swag v0.23.0/go.mod h1:esZ8ITTYEsH1V2trKHjAN8Ai7xHb8RV+YSZ577vPjgQ=
github.com/go-openapi/swag v0.22.4/go.mod h1:UzaqsxGiab7freDnrUUra0MwWfN/q7tE4j+VcZ0yl14=
github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4= github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
github.com/go-playground/locales v0.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8= github.com/go-playground/locales v0.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8=
github.com/go-playground/locales v0.14.0/go.mod h1:sawfccIbzZTqEDETgFXqTho0QybSa7l++s0DH+LDiLs= github.com/go-playground/locales v0.14.0/go.mod h1:sawfccIbzZTqEDETgFXqTho0QybSa7l++s0DH+LDiLs=
@ -519,8 +514,8 @@ github.com/gorilla/context v1.1.1/go.mod h1:kBGZzfjB9CEq2AlWe17Uuf7NDRt0dE0s8S51
github.com/gorilla/websocket v0.0.0-20170926233335-4201258b820c/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ= github.com/gorilla/websocket v0.0.0-20170926233335-4201258b820c/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ=
github.com/gorilla/websocket v1.4.1/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/gorilla/websocket v1.4.1/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
github.com/gorilla/websocket v1.5.0 h1:PPwGk2jz7EePpoHN/+ClbZu8SPxiqlu12wZP/3sWmnc= github.com/gorilla/websocket v1.5.1 h1:gmztn0JnHVt9JZquRuzLw3g4wouNVzKL15iLr/zn/QY=
github.com/gorilla/websocket v1.5.0/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/gorilla/websocket v1.5.1/go.mod h1:x3kM2JMyaluk02fnUJpQuwD2dCS5NDG2ZHL0uE0tcaY=
github.com/gravitational/trace v1.1.16-0.20220114165159-14a9a7dd6aaf h1:C1GPyPJrOlJlIrcaBBiBpDsqZena2Ks8spa5xZqr1XQ= github.com/gravitational/trace v1.1.16-0.20220114165159-14a9a7dd6aaf h1:C1GPyPJrOlJlIrcaBBiBpDsqZena2Ks8spa5xZqr1XQ=
github.com/gravitational/trace v1.1.16-0.20220114165159-14a9a7dd6aaf/go.mod h1:zXqxTI6jXDdKnlf8s+nT+3c8LrwUEy3yNpO4XJL90lA= github.com/gravitational/trace v1.1.16-0.20220114165159-14a9a7dd6aaf/go.mod h1:zXqxTI6jXDdKnlf8s+nT+3c8LrwUEy3yNpO4XJL90lA=
github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs= github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs=
@ -874,8 +869,8 @@ github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1y
github.com/onsi/gomega v1.17.0/go.mod h1:HnhC7FXeEQY45zxNK3PPoIUhzk/80Xly9PcubAlGdZY= github.com/onsi/gomega v1.17.0/go.mod h1:HnhC7FXeEQY45zxNK3PPoIUhzk/80Xly9PcubAlGdZY=
github.com/onsi/gomega v1.18.1/go.mod h1:0q+aL8jAiMXy9hbwj2mr5GziHiwhAIQpFmmtT5hitRs= github.com/onsi/gomega v1.18.1/go.mod h1:0q+aL8jAiMXy9hbwj2mr5GziHiwhAIQpFmmtT5hitRs=
github.com/onsi/gomega v1.19.0/go.mod h1:LY+I3pBVzYsTBU1AnDwOSxaYi9WoWiqgwooUqq9yPro= github.com/onsi/gomega v1.19.0/go.mod h1:LY+I3pBVzYsTBU1AnDwOSxaYi9WoWiqgwooUqq9yPro=
github.com/onsi/gomega v1.31.1 h1:KYppCUK+bUgAZwHOu7EXVBKyQA6ILvOESHkn/tgoqvo= github.com/onsi/gomega v1.32.0 h1:JRYU78fJ1LPxlckP6Txi/EYqJvjtMrDC04/MM5XRHPk=
github.com/onsi/gomega v1.31.1/go.mod h1:y40C95dwAD1Nz36SsEnxvfFe8FFfNxzI5eJ0EYGyAy0= github.com/onsi/gomega v1.32.0/go.mod h1:a4x4gW6Pz2yK1MAmvluYme5lvYTn61afQ2ETw/8n4Lg=
github.com/op/go-logging v0.0.0-20160315200505-970db520ece7/go.mod h1:HzydrMdWErDVzsI23lYNej1Htcns9BCg93Dk0bBINWk= github.com/op/go-logging v0.0.0-20160315200505-970db520ece7/go.mod h1:HzydrMdWErDVzsI23lYNej1Htcns9BCg93Dk0bBINWk=
github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8Oi/yOhh5U= github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8Oi/yOhh5U=
github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM= github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM=
@ -1168,12 +1163,12 @@ github.com/zeebo/errs v1.2.2/go.mod h1:sgbWHsvVuTPHcqJJGQ1WhI5KbWlHYz+2+2C/LSEtC
go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU= go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
go.etcd.io/bbolt v1.3.3/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU= go.etcd.io/bbolt v1.3.3/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
go.etcd.io/etcd v0.0.0-20191023171146-3cf2f69b5738/go.mod h1:dnLIgRNXwCJa5e+c6mIZCrds/GIG4ncV9HhK5PX7jPg= go.etcd.io/etcd v0.0.0-20191023171146-3cf2f69b5738/go.mod h1:dnLIgRNXwCJa5e+c6mIZCrds/GIG4ncV9HhK5PX7jPg=
go.etcd.io/etcd/api/v3 v3.5.9 h1:4wSsluwyTbGGmyjJktOf3wFQoTBIURXHnq9n/G/JQHs= go.etcd.io/etcd/api/v3 v3.5.10 h1:szRajuUUbLyppkhs9K6BRtjY37l66XQQmw7oZRANE4k=
go.etcd.io/etcd/api/v3 v3.5.9/go.mod h1:uyAal843mC8uUVSLWz6eHa/d971iDGnCRpmKd2Z+X8k= go.etcd.io/etcd/api/v3 v3.5.10/go.mod h1:TidfmT4Uycad3NM/o25fG3J07odo4GBB9hoxaodFCtI=
go.etcd.io/etcd/client/pkg/v3 v3.5.9 h1:oidDC4+YEuSIQbsR94rY9gur91UPL6DnxDCIYd2IGsE= go.etcd.io/etcd/client/pkg/v3 v3.5.10 h1:kfYIdQftBnbAq8pUWFXfpuuxFSKzlmM5cSn76JByiT0=
go.etcd.io/etcd/client/pkg/v3 v3.5.9/go.mod h1:y+CzeSmkMpWN2Jyu1npecjB9BBnABxGM4pN8cGuJeL4= go.etcd.io/etcd/client/pkg/v3 v3.5.10/go.mod h1:DYivfIviIuQ8+/lCq4vcxuseg2P2XbHygkKwFo9fc8U=
go.etcd.io/etcd/client/v3 v3.5.9 h1:r5xghnU7CwbUxD/fbUtRyJGaYNfDun8sp/gTr1hew6E= go.etcd.io/etcd/client/v3 v3.5.10 h1:W9TXNZ+oB3MCd/8UjxHTWK5J9Nquw9fQBLJd5ne5/Ao=
go.etcd.io/etcd/client/v3 v3.5.9/go.mod h1:i/Eo5LrZ5IKqpbtpPDuaUnDOUv471oDg8cjQaUr2MbA= go.etcd.io/etcd/client/v3 v3.5.10/go.mod h1:RVeBnDz2PUEZqTpgqwAtUd8nAPf5kjyFyND7P1VkOKc=
go.opencensus.io v0.20.1/go.mod h1:6WKK9ahsWS3RSO+PY9ZHZUfv2irvY6gN279GOPZjmmk= go.opencensus.io v0.20.1/go.mod h1:6WKK9ahsWS3RSO+PY9ZHZUfv2irvY6gN279GOPZjmmk=
go.opencensus.io v0.20.2/go.mod h1:6WKK9ahsWS3RSO+PY9ZHZUfv2irvY6gN279GOPZjmmk= go.opencensus.io v0.20.2/go.mod h1:6WKK9ahsWS3RSO+PY9ZHZUfv2irvY6gN279GOPZjmmk=
go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
@ -1221,18 +1216,15 @@ go.opentelemetry.io/proto/otlp v1.2.0/go.mod h1:gGpR8txAl5M03pDhMC79G6SdqNV26naR
go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
go.uber.org/atomic v1.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ= go.uber.org/atomic v1.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ=
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
go.uber.org/atomic v1.11.0 h1:ZvwS0R+56ePWxUNi+Atn9dWONBPp/AUETXlHW0DxSjE= go.uber.org/atomic v1.11.0 h1:ZvwS0R+56ePWxUNi+Atn9dWONBPp/AUETXlHW0DxSjE=
go.uber.org/atomic v1.11.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0= go.uber.org/atomic v1.11.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0=
go.uber.org/goleak v1.1.11/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ=
go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto=
go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE=
go.uber.org/mock v0.4.0 h1:VcM4ZOtdbR4f6VXfiOpwpVJDL6lCReaZ6mw31wqh7KU= go.uber.org/mock v0.4.0 h1:VcM4ZOtdbR4f6VXfiOpwpVJDL6lCReaZ6mw31wqh7KU=
go.uber.org/mock v0.4.0/go.mod h1:a6FSlNadKUHUa9IP5Vyt1zh4fC7uAwxMutEAscFbkZc= go.uber.org/mock v0.4.0/go.mod h1:a6FSlNadKUHUa9IP5Vyt1zh4fC7uAwxMutEAscFbkZc=
go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0= go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
go.uber.org/multierr v1.3.0/go.mod h1:VgVr7evmIr6uPjLBxg28wmKNXyqE9akIJ5XnfpiKl+4= go.uber.org/multierr v1.3.0/go.mod h1:VgVr7evmIr6uPjLBxg28wmKNXyqE9akIJ5XnfpiKl+4=
go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU=
go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0= go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y= go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
go.uber.org/ratelimit v0.3.0 h1:IdZd9wqvFXnvLvSEBo0KPcGfkoBGNkpTHlrE3Rcjkjw= go.uber.org/ratelimit v0.3.0 h1:IdZd9wqvFXnvLvSEBo0KPcGfkoBGNkpTHlrE3Rcjkjw=
@ -1240,7 +1232,6 @@ go.uber.org/ratelimit v0.3.0/go.mod h1:So5LG7CV1zWpY1sHe+DXTJqQvOx+FFPFaAs2SnoyB
go.uber.org/tools v0.0.0-20190618225709-2cfd321de3ee/go.mod h1:vJERXedbb3MVM5f9Ejo0C68/HhF8uaILCdgjnY+goOA= go.uber.org/tools v0.0.0-20190618225709-2cfd321de3ee/go.mod h1:vJERXedbb3MVM5f9Ejo0C68/HhF8uaILCdgjnY+goOA=
go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q= go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
go.uber.org/zap v1.13.0/go.mod h1:zwrFLgMcdUuIBviXEYEH1YKNaOBnKXsx2IPda5bBwHM= go.uber.org/zap v1.13.0/go.mod h1:zwrFLgMcdUuIBviXEYEH1YKNaOBnKXsx2IPda5bBwHM=
go.uber.org/zap v1.24.0/go.mod h1:2kMP+WWQ8aoFoedH3T2sq6iJ2yDWpHbP0f6MQbS9Gkg=
go.uber.org/zap v1.26.0 h1:sI7k6L95XOKS281NhVKOFCUNIvv9e0w4BF8N3u+tCRo= go.uber.org/zap v1.26.0 h1:sI7k6L95XOKS281NhVKOFCUNIvv9e0w4BF8N3u+tCRo=
go.uber.org/zap v1.26.0/go.mod h1:dtElttAiwGvoJ/vj4IwHBS/gXsEu/pZ50mUIRWuG0so= go.uber.org/zap v1.26.0/go.mod h1:dtElttAiwGvoJ/vj4IwHBS/gXsEu/pZ50mUIRWuG0so=
golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
@ -1280,8 +1271,8 @@ golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL
golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8=
golang.org/x/exp v0.0.0-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek= golang.org/x/exp v0.0.0-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek=
golang.org/x/exp v0.0.0-20191030013958-a1ab85dbe136/go.mod h1:JXzH8nQsPlswgeRAPE3MuO9GYsAcnJvJ4vnMwN/5qkY= golang.org/x/exp v0.0.0-20191030013958-a1ab85dbe136/go.mod h1:JXzH8nQsPlswgeRAPE3MuO9GYsAcnJvJ4vnMwN/5qkY=
golang.org/x/exp v0.0.0-20240404231335-c0f41cb1a7a0 h1:985EYyeCOxTpcgOTJpflJUwOeEz0CQOdPt73OzpE9F8= golang.org/x/exp v0.0.0-20240416160154-fe59bbe5cc7f h1:99ci1mjWVBWwJiEKYY6jWa4d2nTQVIEhZIptnrVb1XY=
golang.org/x/exp v0.0.0-20240404231335-c0f41cb1a7a0/go.mod h1:/lliqkxwWAhPjf5oSOIJup2XcqJaw8RGS6k3TGEc7GI= golang.org/x/exp v0.0.0-20240416160154-fe59bbe5cc7f/go.mod h1:/lliqkxwWAhPjf5oSOIJup2XcqJaw8RGS6k3TGEc7GI=
golang.org/x/image v0.0.0-20180708004352-c73c2afc3b81/go.mod h1:ux5Hcp/YLpHSI86hEcLt0YII63i6oz57MZXIpbrjZUs= golang.org/x/image v0.0.0-20180708004352-c73c2afc3b81/go.mod h1:ux5Hcp/YLpHSI86hEcLt0YII63i6oz57MZXIpbrjZUs=
golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js=
golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0=
@ -1507,7 +1498,6 @@ golang.org/x/tools v0.0.0-20200918232735-d647fc253266/go.mod h1:z6u4i615ZeAfBE4X
golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
golang.org/x/tools v0.0.0-20210114065538-d78b04bdf963/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20210114065538-d78b04bdf963/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk=
golang.org/x/tools v0.1.6-0.20210726203631-07bc1bf47fb2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.6-0.20210726203631-07bc1bf47fb2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk=
golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU=
@ -1517,8 +1507,6 @@ golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8T
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
gomodules.xyz/jsonpatch/v2 v2.4.0 h1:Ci3iUJyx9UeRx7CeFN8ARgGbkESwJK+KB9lLcWxY/Zw=
gomodules.xyz/jsonpatch/v2 v2.4.0/go.mod h1:AH3dM2RI6uoBZxn3LVrfvJ3E0/9dG4cSrbuBJT4moAY=
gonum.org/v1/gonum v0.0.0-20180816165407-929014505bf4/go.mod h1:Y+Yx5eoAFn32cQvJDxZx5Dpnq+c3wtXuadVZAcxbbBo= gonum.org/v1/gonum v0.0.0-20180816165407-929014505bf4/go.mod h1:Y+Yx5eoAFn32cQvJDxZx5Dpnq+c3wtXuadVZAcxbbBo=
gonum.org/v1/gonum v0.8.2 h1:CCXrcPKiGGotvnN6jfUsKk4rRqm7q09/YbKb5xCEvtM= gonum.org/v1/gonum v0.8.2 h1:CCXrcPKiGGotvnN6jfUsKk4rRqm7q09/YbKb5xCEvtM=
gonum.org/v1/gonum v0.8.2/go.mod h1:oe/vMfY3deqTw+1EZJhuvEW2iwGF1bW9wwu7XCu0+v0= gonum.org/v1/gonum v0.8.2/go.mod h1:oe/vMfY3deqTw+1EZJhuvEW2iwGF1bW9wwu7XCu0+v0=
@ -1650,20 +1638,20 @@ honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWh
honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg= honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg=
k8s.io/api v0.29.2 h1:hBC7B9+MU+ptchxEqTNW2DkUosJpp1P+Wn6YncZ474A= k8s.io/api v0.30.0 h1:siWhRq7cNjy2iHssOB9SCGNCl2spiF1dO3dABqZ8niA=
k8s.io/api v0.29.2/go.mod h1:sdIaaKuU7P44aoyyLlikSLayT6Vb7bvJNCX105xZXY0= k8s.io/api v0.30.0/go.mod h1:OPlaYhoHs8EQ1ql0R/TsUgaRPhpKNxIMrKQfWUp8QSE=
k8s.io/apiextensions-apiserver v0.28.3 h1:Od7DEnhXHnHPZG+W9I97/fSQkVpVPQx2diy+2EtmY08= k8s.io/apiextensions-apiserver v0.30.0 h1:jcZFKMqnICJfRxTgnC4E+Hpcq8UEhT8B2lhBcQ+6uAs=
k8s.io/apiextensions-apiserver v0.28.3/go.mod h1:NE1XJZ4On0hS11aWWJUTNkmVB03j9LM7gJSisbRt8Lc= k8s.io/apiextensions-apiserver v0.30.0/go.mod h1:N9ogQFGcrbWqAY9p2mUAL5mGxsLqwgtUce127VtRX5Y=
k8s.io/apimachinery v0.29.2 h1:EWGpfJ856oj11C52NRCHuU7rFDwxev48z+6DSlGNsV8= k8s.io/apimachinery v0.30.0 h1:qxVPsyDM5XS96NIh9Oj6LavoVFYff/Pon9cZeDIkHHA=
k8s.io/apimachinery v0.29.2/go.mod h1:6HVkd1FwxIagpYrHSwJlQqZI3G9LfYWRPAkUvLnXTKU= k8s.io/apimachinery v0.30.0/go.mod h1:iexa2somDaxdnj7bha06bhb43Zpa6eWH8N8dbqVjTUc=
k8s.io/client-go v0.29.2 h1:FEg85el1TeZp+/vYJM7hkDlSTFZ+c5nnK44DJ4FyoRg= k8s.io/client-go v0.30.0 h1:sB1AGGlhY/o7KCyCEQ0bPWzYDL0pwOZO4vAtTSh/gJQ=
k8s.io/client-go v0.29.2/go.mod h1:knlvFZE58VpqbQpJNbCbctTVXcd35mMyAAwBdpt4jrA= k8s.io/client-go v0.30.0/go.mod h1:g7li5O5256qe6TYdAMyX/otJqMhIiGgTapdLchhmOaY=
k8s.io/klog/v2 v2.110.1 h1:U/Af64HJf7FcwMcXyKm2RPM22WZzyR7OSpYj5tg3cL0= k8s.io/klog/v2 v2.120.1 h1:QXU6cPEOIslTGvZaXvFWiP9VKyeet3sawzTOvdXb4Vw=
k8s.io/klog/v2 v2.110.1/go.mod h1:YGtd1984u+GgbuZ7e08/yBuAfKLSO0+uR1Fhi6ExXjo= k8s.io/klog/v2 v2.120.1/go.mod h1:3Jpz1GvMt720eyJH1ckRHK1EDfpxISzJ7I9OYgaDtPE=
k8s.io/kube-openapi v0.0.0-20231010175941-2dd684a91f00 h1:aVUu9fTY98ivBPKR9Y5w/AuzbMm96cd3YHRTU83I780= k8s.io/kube-openapi v0.0.0-20240423202451-8948a665c108 h1:Q8Z7VlGhcJgBHJHYugJ/K/7iB8a2eSxCyxdVjJp+lLY=
k8s.io/kube-openapi v0.0.0-20231010175941-2dd684a91f00/go.mod h1:AsvuZPBlUDVuCdzJ87iajxtXuR9oktsTctW/R9wwouA= k8s.io/kube-openapi v0.0.0-20240423202451-8948a665c108/go.mod h1:yD4MZYeKMBwQKVht279WycxKyM84kkAx2DPrTXaeb98=
k8s.io/utils v0.0.0-20230726121419-3b25d923346b h1:sgn3ZU783SCgtaSJjpcVVlRqd6GSnlTLKgpAAttJvpI= k8s.io/utils v0.0.0-20240423183400-0849a56e8f22 h1:ao5hUqGhsqdm+bYbjH/pRkCs0unBGe9UyDahzs9zQzQ=
k8s.io/utils v0.0.0-20230726121419-3b25d923346b/go.mod h1:OLgZIPagt7ERELqWJFomSt595RzquPNLL48iOWgYOg0= k8s.io/utils v0.0.0-20240423183400-0849a56e8f22/go.mod h1:OLgZIPagt7ERELqWJFomSt595RzquPNLL48iOWgYOg0=
mvdan.cc/xurls/v2 v2.5.0 h1:lyBNOm8Wo71UknhUs4QTFUNNMyxy2JEIaKKo0RWOh+8= mvdan.cc/xurls/v2 v2.5.0 h1:lyBNOm8Wo71UknhUs4QTFUNNMyxy2JEIaKKo0RWOh+8=
mvdan.cc/xurls/v2 v2.5.0/go.mod h1:yQgaGQ1rFtJUzkmKiHYSSfuQxqfYmd//X6PxvholpeE= mvdan.cc/xurls/v2 v2.5.0/go.mod h1:yQgaGQ1rFtJUzkmKiHYSSfuQxqfYmd//X6PxvholpeE=
nhooyr.io/websocket v1.8.7 h1:usjR2uOr/zjjkVMy0lW+PPohFok7PCow5sDjLgX4P4g= nhooyr.io/websocket v1.8.7 h1:usjR2uOr/zjjkVMy0lW+PPohFok7PCow5sDjLgX4P4g=
@ -1671,10 +1659,10 @@ nhooyr.io/websocket v1.8.7/go.mod h1:B70DZP8IakI65RVQ51MsWP/8jndNma26DVA/nFSCgW0
nullprogram.com/x/optparse v1.0.0/go.mod h1:KdyPE+Igbe0jQUrVfMqDMeJQIJZEuyV7pjYmp6pbG50= nullprogram.com/x/optparse v1.0.0/go.mod h1:KdyPE+Igbe0jQUrVfMqDMeJQIJZEuyV7pjYmp6pbG50=
rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4= rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4=
sigs.k8s.io/controller-runtime v0.16.3 h1:2TuvuokmfXvDUamSx1SuAOO3eTyye+47mJCigwG62c4= sigs.k8s.io/controller-runtime v0.18.0 h1:Z7jKuX784TQSUL1TIyeuF7j8KXZ4RtSX0YgtjKcSTME=
sigs.k8s.io/controller-runtime v0.16.3/go.mod h1:j7bialYoSn142nv9sCOJmQgDXQXxnroFU4VnX/brVJ0= sigs.k8s.io/controller-runtime v0.18.0/go.mod h1:tuAt1+wbVsXIT8lPtk5RURxqAnq7xkpv2Mhttslg7Hw=
sigs.k8s.io/gateway-api v1.0.0 h1:iPTStSv41+d9p0xFydll6d7f7MOBGuqXM6p2/zVYMAs= sigs.k8s.io/gateway-api v1.1.0 h1:DsLDXCi6jR+Xz8/xd0Z1PYl2Pn0TyaFMOPPZIj4inDM=
sigs.k8s.io/gateway-api v1.0.0/go.mod h1:4cUgr0Lnp5FZ0Cdq8FdRwCvpiWws7LVhLHGIudLlf4c= sigs.k8s.io/gateway-api v1.1.0/go.mod h1:ZH4lHrL2sDi0FHZ9jjneb8kKnGzFWyrTya35sWUTrRs=
sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd h1:EDPBXCAspyGV4jQlpZSudPeMmr1bNJefnuqLsRAsHZo= sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd h1:EDPBXCAspyGV4jQlpZSudPeMmr1bNJefnuqLsRAsHZo=
sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd/go.mod h1:B8JuhiUyNFVKdsE8h686QcCxMaH6HrOAZj4vswFpcB0= sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd/go.mod h1:B8JuhiUyNFVKdsE8h686QcCxMaH6HrOAZj4vswFpcB0=
sigs.k8s.io/structured-merge-diff/v4 v4.4.1 h1:150L+0vs/8DA78h1u02ooW1/fFq/Lwr+sGiqlzvrtq4= sigs.k8s.io/structured-merge-diff/v4 v4.4.1 h1:150L+0vs/8DA78h1u02ooW1/fFq/Lwr+sGiqlzvrtq4=

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

View file

@ -15,12 +15,19 @@ rules:
- "" - ""
resources: resources:
- services - services
- endpoints
- secrets - secrets
verbs: verbs:
- get - get
- list - list
- watch - watch
- apiGroups:
- discovery.k8s.io
resources:
- endpointslices
verbs:
- get
- list
- watch
- apiGroups: - apiGroups:
- gateway.networking.k8s.io - gateway.networking.k8s.io
resources: resources:

View file

@ -4,6 +4,7 @@ import (
"context" "context"
"fmt" "fmt"
"io" "io"
"io/fs"
"os" "os"
"path/filepath" "path/filepath"
"slices" "slices"
@ -18,6 +19,7 @@ import (
"github.com/traefik/traefik/v3/integration/try" "github.com/traefik/traefik/v3/integration/try"
"github.com/traefik/traefik/v3/pkg/version" "github.com/traefik/traefik/v3/pkg/version"
"gopkg.in/yaml.v3" "gopkg.in/yaml.v3"
apiextensionsv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
"k8s.io/apimachinery/pkg/util/sets" "k8s.io/apimachinery/pkg/util/sets"
kclientset "k8s.io/client-go/kubernetes" kclientset "k8s.io/client-go/kubernetes"
"k8s.io/client-go/tools/clientcmd" "k8s.io/client-go/tools/clientcmd"
@ -27,10 +29,12 @@ import (
gatev1 "sigs.k8s.io/gateway-api/apis/v1" gatev1 "sigs.k8s.io/gateway-api/apis/v1"
gatev1alpha2 "sigs.k8s.io/gateway-api/apis/v1alpha2" gatev1alpha2 "sigs.k8s.io/gateway-api/apis/v1alpha2"
gatev1beta1 "sigs.k8s.io/gateway-api/apis/v1beta1" gatev1beta1 "sigs.k8s.io/gateway-api/apis/v1beta1"
conformanceV1alpha1 "sigs.k8s.io/gateway-api/conformance/apis/v1alpha1" "sigs.k8s.io/gateway-api/conformance"
v1 "sigs.k8s.io/gateway-api/conformance/apis/v1"
"sigs.k8s.io/gateway-api/conformance/tests" "sigs.k8s.io/gateway-api/conformance/tests"
"sigs.k8s.io/gateway-api/conformance/utils/config" "sigs.k8s.io/gateway-api/conformance/utils/config"
ksuite "sigs.k8s.io/gateway-api/conformance/utils/suite" ksuite "sigs.k8s.io/gateway-api/conformance/utils/suite"
"sigs.k8s.io/gateway-api/pkg/features"
) )
const ( const (
@ -84,7 +88,7 @@ func (s *K8sConformanceSuite) SetupSuite() {
s.k3sContainer, err = k3s.RunContainer(ctx, s.k3sContainer, err = k3s.RunContainer(ctx,
testcontainers.WithImage(k3sImage), testcontainers.WithImage(k3sImage),
k3s.WithManifest("./fixtures/k8s-conformance/00-experimental-v1.0.0.yml"), k3s.WithManifest("./fixtures/k8s-conformance/00-experimental-v1.1.0.yml"),
k3s.WithManifest("./fixtures/k8s-conformance/01-rbac.yml"), k3s.WithManifest("./fixtures/k8s-conformance/01-rbac.yml"),
k3s.WithManifest("./fixtures/k8s-conformance/02-traefik.yml"), k3s.WithManifest("./fixtures/k8s-conformance/02-traefik.yml"),
network.WithNetwork(nil, s.network), network.WithNetwork(nil, s.network),
@ -122,15 +126,19 @@ func (s *K8sConformanceSuite) SetupSuite() {
s.T().Fatalf("Error initializing Kubernetes REST client: %v", err) s.T().Fatalf("Error initializing Kubernetes REST client: %v", err)
} }
if err = gatev1alpha2.AddToScheme(s.kubeClient.Scheme()); err != nil { if err = gatev1alpha2.Install(s.kubeClient.Scheme()); err != nil {
s.T().Fatal(err) s.T().Fatal(err)
} }
if err = gatev1beta1.AddToScheme(s.kubeClient.Scheme()); err != nil { if err = gatev1beta1.Install(s.kubeClient.Scheme()); err != nil {
s.T().Fatal(err) s.T().Fatal(err)
} }
if err = gatev1.AddToScheme(s.kubeClient.Scheme()); err != nil { if err = gatev1.Install(s.kubeClient.Scheme()); err != nil {
s.T().Fatal(err)
}
if err = apiextensionsv1.AddToScheme(s.kubeClient.Scheme()); err != nil {
s.T().Fatal(err) s.T().Fatal(err)
} }
} }
@ -169,81 +177,54 @@ func (s *K8sConformanceSuite) TestK8sGatewayAPIConformance() {
err = try.GetRequest("http://"+k3sContainerIP+":9000/api/entrypoints", 10*time.Second, try.BodyContains(`"name":"web"`)) err = try.GetRequest("http://"+k3sContainerIP+":9000/api/entrypoints", 10*time.Second, try.BodyContains(`"name":"web"`))
require.NoError(s.T(), err) require.NoError(s.T(), err)
opts := ksuite.Options{ cSuite, err := ksuite.NewConformanceTestSuite(ksuite.ConformanceOptions{
Client: s.kubeClient, Client: s.kubeClient,
Clientset: s.clientSet, Clientset: s.clientSet,
GatewayClassName: "traefik", GatewayClassName: "traefik",
Debug: true, Debug: true,
CleanupBaseResources: true, CleanupBaseResources: true,
TimeoutConfig: config.TimeoutConfig{ TimeoutConfig: config.DefaultTimeoutConfig(),
CreateTimeout: 5 * time.Second, ManifestFS: []fs.FS{&conformance.Manifests},
DeleteTimeout: 5 * time.Second,
GetTimeout: 5 * time.Second,
GatewayMustHaveAddress: 5 * time.Second,
GatewayMustHaveCondition: 5 * time.Second,
GatewayStatusMustHaveListeners: 10 * time.Second,
GatewayListenersMustHaveCondition: 5 * time.Second,
GWCMustBeAccepted: 60 * time.Second, // Pod creation in k3s cluster can be long.
HTTPRouteMustNotHaveParents: 5 * time.Second,
HTTPRouteMustHaveCondition: 5 * time.Second,
TLSRouteMustHaveCondition: 5 * time.Second,
RouteMustHaveParents: 5 * time.Second,
ManifestFetchTimeout: 5 * time.Second,
MaxTimeToConsistency: 5 * time.Second,
NamespacesMustBeReady: 60 * time.Second, // Pod creation in k3s cluster can be long.
RequestTimeout: 5 * time.Second,
LatestObservedGenerationSet: 5 * time.Second,
RequiredConsecutiveSuccesses: 0,
},
SupportedFeatures: sets.New(ksuite.SupportGateway,
ksuite.SupportGatewayPort8080,
ksuite.SupportHTTPRoute,
ksuite.SupportHTTPRouteQueryParamMatching,
ksuite.SupportHTTPRouteMethodMatching,
ksuite.SupportHTTPRoutePortRedirect,
ksuite.SupportHTTPRouteSchemeRedirect,
ksuite.SupportHTTPRouteHostRewrite,
ksuite.SupportHTTPRoutePathRewrite,
ksuite.SupportHTTPRoutePathRedirect,
),
ExemptFeatures: sets.New(
ksuite.SupportHTTPRouteRequestTimeout,
ksuite.SupportHTTPRouteBackendTimeout,
ksuite.SupportHTTPRouteResponseHeaderModification,
ksuite.SupportHTTPRouteRequestMirror,
ksuite.SupportHTTPRouteRequestMultipleMirrors,
),
EnableAllSupportedFeatures: false, EnableAllSupportedFeatures: false,
RunTest: *k8sConformanceRunTest, RunTest: *k8sConformanceRunTest,
// Until the feature are all supported, following tests are skipped. Implementation: v1.Implementation{
SkipTests: []string{
tests.HTTPRouteMethodMatching.ShortName,
tests.HTTPRouteQueryParamMatching.ShortName,
},
}
cSuite, err := ksuite.NewExperimentalConformanceTestSuite(ksuite.ExperimentalConformanceOptions{
Options: opts,
Implementation: conformanceV1alpha1.Implementation{
Organization: "traefik", Organization: "traefik",
Project: "traefik", Project: "traefik",
URL: "https://traefik.io/", URL: "https://traefik.io/",
Version: version.Version, Version: version.Version,
Contact: []string{"@traefik/maintainers"}, Contact: []string{"@traefik/maintainers"},
}, },
ConformanceProfiles: sets.New(ksuite.HTTPConformanceProfileName), ConformanceProfiles: sets.New(ksuite.GatewayHTTPConformanceProfileName),
SupportedFeatures: sets.New(
features.SupportGateway,
features.SupportGatewayPort8080,
features.SupportHTTPRoute,
features.SupportHTTPRouteQueryParamMatching,
features.SupportHTTPRouteMethodMatching,
features.SupportHTTPRoutePortRedirect,
features.SupportHTTPRouteSchemeRedirect,
features.SupportHTTPRouteHostRewrite,
features.SupportHTTPRoutePathRewrite,
features.SupportHTTPRoutePathRedirect,
),
ExemptFeatures: sets.New(
features.SupportHTTPRouteRequestTimeout,
features.SupportHTTPRouteBackendTimeout,
features.SupportHTTPRouteResponseHeaderModification,
features.SupportHTTPRouteRequestMirror,
features.SupportHTTPRouteRequestMultipleMirrors,
),
}) })
require.NoError(s.T(), err) require.NoError(s.T(), err)
cSuite.Setup(s.T()) cSuite.Setup(s.T(), tests.ConformanceTests)
err = cSuite.Run(s.T(), tests.ConformanceTests) err = cSuite.Run(s.T(), tests.ConformanceTests)
require.NoError(s.T(), err) require.NoError(s.T(), err)
report, err := cSuite.Report() report, err := cSuite.Report()
require.NoError(s.T(), err, "failed generating conformance report") require.NoError(s.T(), err, "failed generating conformance report")
report.GatewayAPIVersion = "1.0.0"
rawReport, err := yaml.Marshal(report) rawReport, err := yaml.Marshal(report)
require.NoError(s.T(), err) require.NoError(s.T(), err)
s.T().Logf("Conformance report:\n%s", string(rawReport)) s.T().Logf("Conformance report:\n%s", string(rawReport))

View file

@ -1,7 +1,7 @@
version: "3.8" version: "3.8"
services: services:
server: server:
image: rancher/k3s:v1.20.15-k3s1 image: rancher/k3s:v1.21.14-k3s1
privileged: true privileged: true
command: command:
- server - server
@ -26,7 +26,7 @@ services:
- ./fixtures/k8s:/var/lib/rancher/k3s/server/manifests - ./fixtures/k8s:/var/lib/rancher/k3s/server/manifests
node: node:
image: rancher/k3s:v1.20.15-k3s1 image: rancher/k3s:v1.21.14-k3s1
privileged: true privileged: true
environment: environment:
K3S_TOKEN: somethingtotallyrandom K3S_TOKEN: somethingtotallyrandom

View file

@ -459,6 +459,7 @@ func (h *http) deprecationNotice(logger zerolog.Logger) bool {
type experimental struct { type experimental struct {
HTTP3 *bool `json:"http3,omitempty" toml:"http3,omitempty" yaml:"http3,omitempty"` HTTP3 *bool `json:"http3,omitempty" toml:"http3,omitempty" yaml:"http3,omitempty"`
KubernetesGateway *bool `json:"kubernetesGateway,omitempty" toml:"kubernetesGateway,omitempty" yaml:"kubernetesGateway,omitempty"`
} }
func (e *experimental) deprecationNotice(logger zerolog.Logger) bool { func (e *experimental) deprecationNotice(logger zerolog.Logger) bool {
@ -469,11 +470,17 @@ func (e *experimental) deprecationNotice(logger zerolog.Logger) bool {
if e.HTTP3 != nil { if e.HTTP3 != nil {
logger.Error().Msg("HTTP3 is not an experimental feature in v3 and the associated enablement has been removed." + logger.Error().Msg("HTTP3 is not an experimental feature in v3 and the associated enablement has been removed." +
"Please remove its usage from the static configuration for Traefik to start." + "Please remove its usage from the static configuration for Traefik to start." +
"For more information please read the migration guide: https://doc.traefik.io/traefik/v3.0/migration/v2-to-v3/#http3-experimental-configuration") "For more information please read the migration guide: https://doc.traefik.io/traefik/v3.0/migration/v2-to-v3-details/#http3")
return true return true
} }
if e.KubernetesGateway != nil {
logger.Error().Msg("KubernetesGateway provider is not an experimental feature starting with v3.1." +
"Please remove its usage from the static configuration." +
"For more information please read the migration guide: https://doc.traefik.io/traefik/v3.0/migration/v3/#gateway-api-kubernetesgateway-provider")
}
return false return false
} }

View file

@ -19,6 +19,7 @@ func TestDeprecationNotice(t *testing.T) {
tests := []struct { tests := []struct {
desc string desc string
config configuration config configuration
wantCompatible bool
}{ }{
{ {
desc: "Docker provider swarmMode option is incompatible", desc: "Docker provider swarmMode option is incompatible",
@ -196,6 +197,15 @@ func TestDeprecationNotice(t *testing.T) {
}, },
}, },
}, },
{
desc: "Experimental KubernetesGateway enablement configuration is compatible",
config: configuration{
Experimental: &experimental{
KubernetesGateway: ptr(true),
},
},
wantCompatible: true,
},
{ {
desc: "Tracing SpanNameLimit option is incompatible", desc: "Tracing SpanNameLimit option is incompatible",
config: configuration{ config: configuration{
@ -278,7 +288,8 @@ func TestDeprecationNotice(t *testing.T) {
}) })
logger := log.With().Logger().Hook(testHook) logger := log.With().Logger().Hook(testHook)
assert.True(t, test.config.deprecationNotice(logger))
assert.Equal(t, !test.wantCompatible, test.config.deprecationNotice(logger))
assert.True(t, gotLog) assert.True(t, gotLog)
assert.Equal(t, zerolog.ErrorLevel, gotLevel) assert.Equal(t, zerolog.ErrorLevel, gotLevel)
}) })

View file

@ -7,5 +7,6 @@ type Experimental struct {
Plugins map[string]plugins.Descriptor `description:"Plugins configuration." json:"plugins,omitempty" toml:"plugins,omitempty" yaml:"plugins,omitempty" export:"true"` Plugins map[string]plugins.Descriptor `description:"Plugins configuration." json:"plugins,omitempty" toml:"plugins,omitempty" yaml:"plugins,omitempty" export:"true"`
LocalPlugins map[string]plugins.LocalDescriptor `description:"Local plugins configuration." json:"localPlugins,omitempty" toml:"localPlugins,omitempty" yaml:"localPlugins,omitempty" export:"true"` LocalPlugins map[string]plugins.LocalDescriptor `description:"Local plugins configuration." json:"localPlugins,omitempty" toml:"localPlugins,omitempty" yaml:"localPlugins,omitempty" export:"true"`
KubernetesGateway bool `description:"Allow the Kubernetes gateway api provider usage." json:"kubernetesGateway,omitempty" toml:"kubernetesGateway,omitempty" yaml:"kubernetesGateway,omitempty" export:"true"` // Deprecated: KubernetesGateway provider is not an experimental feature starting with v3.1. Please remove its usage from the static configuration.
KubernetesGateway bool `description:"(Deprecated) Allow the Kubernetes gateway api provider usage." json:"kubernetesGateway,omitempty" toml:"kubernetesGateway,omitempty" yaml:"kubernetesGateway,omitempty" export:"true"`
} }

View file

@ -17,9 +17,11 @@ import (
"github.com/traefik/traefik/v3/pkg/types" "github.com/traefik/traefik/v3/pkg/types"
"github.com/traefik/traefik/v3/pkg/version" "github.com/traefik/traefik/v3/pkg/version"
corev1 "k8s.io/api/core/v1" corev1 "k8s.io/api/core/v1"
discoveryv1 "k8s.io/api/discovery/v1"
kerror "k8s.io/apimachinery/pkg/api/errors" kerror "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/labels" "k8s.io/apimachinery/pkg/labels"
"k8s.io/apimachinery/pkg/selection"
kinformers "k8s.io/client-go/informers" kinformers "k8s.io/client-go/informers"
kclientset "k8s.io/client-go/kubernetes" kclientset "k8s.io/client-go/kubernetes"
"k8s.io/client-go/rest" "k8s.io/client-go/rest"
@ -46,7 +48,7 @@ type Client interface {
GetTLSStores() []*traefikv1alpha1.TLSStore GetTLSStores() []*traefikv1alpha1.TLSStore
GetService(namespace, name string) (*corev1.Service, bool, error) GetService(namespace, name string) (*corev1.Service, bool, error)
GetSecret(namespace, name string) (*corev1.Secret, bool, error) GetSecret(namespace, name string) (*corev1.Secret, bool, error)
GetEndpoints(namespace, name string) (*corev1.Endpoints, bool, error) GetEndpointSlicesForService(namespace, serviceName string) ([]*discoveryv1.EndpointSlice, error)
GetNodes() ([]*corev1.Node, bool, error) GetNodes() ([]*corev1.Node, bool, error)
} }
@ -219,7 +221,7 @@ func (c *clientWrapper) WatchAll(namespaces []string, stopCh <-chan struct{}) (<
if err != nil { if err != nil {
return nil, err return nil, err
} }
_, err = factoryKube.Core().V1().Endpoints().Informer().AddEventHandler(eventHandler) _, err = factoryKube.Discovery().V1().EndpointSlices().Informer().AddEventHandler(eventHandler)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -444,15 +446,20 @@ func (c *clientWrapper) GetService(namespace, name string) (*corev1.Service, boo
return service, exist, err return service, exist, err
} }
// GetEndpoints returns the named endpoints from the given namespace. // GetEndpointSlicesForService returns the EndpointSlices for the given service name in the given namespace.
func (c *clientWrapper) GetEndpoints(namespace, name string) (*corev1.Endpoints, bool, error) { func (c *clientWrapper) GetEndpointSlicesForService(namespace, serviceName string) ([]*discoveryv1.EndpointSlice, error) {
if !c.isWatchedNamespace(namespace) { if !c.isWatchedNamespace(namespace) {
return nil, false, fmt.Errorf("failed to get endpoints %s/%s: namespace is not within watched namespaces", namespace, name) return nil, fmt.Errorf("failed to get endpointslices for service %s/%s: namespace is not within watched namespaces", namespace, serviceName)
} }
endpoint, err := c.factoriesKube[c.lookupNamespace(namespace)].Core().V1().Endpoints().Lister().Endpoints(namespace).Get(name) serviceLabelRequirement, err := labels.NewRequirement(discoveryv1.LabelServiceName, selection.Equals, []string{serviceName})
exist, err := translateNotFoundError(err) if err != nil {
return endpoint, exist, err return nil, fmt.Errorf("failed to create service label selector requirement: %w", err)
}
serviceSelector := labels.NewSelector()
serviceSelector = serviceSelector.Add(*serviceLabelRequirement)
return c.factoriesKube[c.lookupNamespace(namespace)].Discovery().V1().EndpointSlices().Lister().EndpointSlices(namespace).List(serviceSelector)
} }
// GetSecret returns the named secret from the given namespace. // GetSecret returns the named secret from the given namespace.

View file

@ -13,19 +13,24 @@ spec:
task: whoami task: whoami
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami name: whoami-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: web - name: web
port: 80 port: 80
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -43,19 +48,24 @@ spec:
task: whoami2 task: whoami2
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami2 name: whoami2-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami2
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -74,19 +84,24 @@ spec:
task: whoami2 task: whoami2
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitls name: whoamitls-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamitls
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.5
- ip: 10.10.0.6
ports:
- name: websecure - name: websecure
port: 8443 port: 8443
endpoints:
- addresses:
- 10.10.0.5
- 10.10.0.6
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -99,25 +114,29 @@ spec:
ports: ports:
- name: websecure2 - name: websecure2
port: 8443 port: 8443
scheme: https
selector: selector:
app: traefiklabs app: traefiklabs
task: whoami3 task: whoami3
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami3 name: whoami3-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami3
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.7
- ip: 10.10.0.8
ports:
- name: websecure2 - name: websecure2
port: 8443 port: 8443
endpoints:
- addresses:
- 10.10.0.7
- 10.10.0.8
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -135,18 +154,23 @@ spec:
task: whoami-ipv6 task: whoami-ipv6
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami-ipv6 name: whoami-ipv6-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami-ipv6
subsets: addressType: IPv6
- addresses: ports:
- ip: "2001:db8:85a3:8d3:1319:8a2e:370:7348"
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- "2001:db8:85a3:8d3:1319:8a2e:370:7348"
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -216,25 +240,30 @@ spec:
task: whoami task: whoami
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami-svc name: whoami-svc-abc
namespace: cross-ns namespace: cross-ns
labels:
kubernetes.io/service-name: whoami-svc
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: web - name: web
port: 80 port: 80
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
kind: Service kind: Service
metadata: metadata:
name: whoami-without-endpoints-subsets name: whoami-without-endpointslice-endpoints
namespace: default namespace: default
spec: spec:
@ -247,11 +276,16 @@ spec:
task: whoami task: whoami
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami-without-endpoints-subsets name: whoami-without-endpointslice-endpoints-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami-without-endpointslice-endpoints
addressType: IPv4
endpoints: []
--- ---
apiVersion: v1 apiVersion: v1

View file

@ -13,19 +13,24 @@ spec:
task: whoamitcp task: whoamitcp
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitcp name: whoamitcp-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamitcp
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: myapp - name: myapp
port: 8000 port: 8000
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -43,19 +48,24 @@ spec:
task: whoamitcp2 task: whoamitcp2
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitcp2 name: whoamitcp2-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamitcp2
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: myapp2 - name: myapp2
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -73,19 +83,24 @@ spec:
task: whoamitcptls2 task: whoamitcptls2
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitcptls name: whoamitcptls-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamitcptls
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.5
- ip: 10.10.0.6
ports:
- name: websecure - name: websecure
port: 443 port: 443
endpoints:
- addresses:
- 10.10.0.5
- 10.10.0.6
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -103,34 +118,44 @@ spec:
task: whoamitcp3 task: whoamitcp3
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitcp3 name: whoamitcp3-abc
namespace: ns3 namespace: ns3
labels:
kubernetes.io/service-name: whoamitcp3
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.7
- ip: 10.10.0.8
ports:
- name: myapp3 - name: myapp3
port: 8083 port: 8083
endpoints:
- addresses:
- 10.10.0.7
- 10.10.0.8
conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitcp3 name: whoamitcp3-abc
namespace: ns4 namespace: ns4
labels:
kubernetes.io/service-name: whoamitcp3
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.9
- ip: 10.10.0.10
ports:
- name: myapp4 - name: myapp4
port: 8084 port: 8084
endpoints:
- addresses:
- 10.10.0.9
- 10.10.0.10
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -148,19 +173,24 @@ spec:
task: whoamitcp-ipv6 task: whoamitcp-ipv6
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitcp-ipv6 name: whoamitcp-ipv6-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamitcp-ipv6
subsets: addressType: IPv6
- addresses: ports:
- ip: "fd00:10:244:0:1::3"
- ip: "2001:db8:85a3:8d3:1319:8a2e:370:7348"
ports:
- name: myapp-ipv6 - name: myapp-ipv6
port: 8080 port: 8080
endpoints:
- addresses:
- "fd00:10:244:0:1::3"
- "2001:db8:85a3:8d3:1319:8a2e:370:7348"
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -212,25 +242,30 @@ spec:
task: whoamitcp task: whoamitcp
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitcp-cross-ns name: whoamitcp-cross-ns-abc
namespace: cross-ns namespace: cross-ns
labels:
kubernetes.io/service-name: whoamitcp-cross-ns
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: myapp - name: myapp
port: 8000 port: 8000
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
kind: Service kind: Service
metadata: metadata:
name: whoamitcp-without-endpoints-subsets name: whoamitcp-without-endpointslice-endpoints
namespace: default namespace: default
spec: spec:
@ -243,11 +278,16 @@ spec:
task: whoamitcp task: whoamitcp
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitcp-without-endpoints-subsets name: whoamitcp-without-endpointslice-endpoints-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamitcp-without-endpointslice-endpoints
addressType: IPv4
endpoints: []
--- ---
apiVersion: v1 apiVersion: v1

View file

@ -11,5 +11,5 @@ spec:
routes: routes:
- match: HostSNI(`foo.com`) - match: HostSNI(`foo.com`)
services: services:
- name: whoamitcp-without-endpoints-subsets - name: whoamitcp-without-endpointslice-endpoints
port: 8000 port: 8000

View file

@ -13,19 +13,24 @@ spec:
task: whoamiudp task: whoamiudp
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamiudp name: whoamiudp-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamiudp
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: myapp - name: myapp
port: 8000 port: 8000
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -43,19 +48,24 @@ spec:
task: whoamiudp2 task: whoamiudp2
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamiudp2 name: whoamiudp2-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamiudp2
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: myapp2 - name: myapp2
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -73,34 +83,44 @@ spec:
task: whoamiudp3 task: whoamiudp3
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamiudp3 name: whoamiudp3-abc
namespace: ns3 namespace: ns3
labels:
kubernetes.io/service-name: whoamiudp3
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.7
- ip: 10.10.0.8
ports:
- name: myapp3 - name: myapp3
port: 8083 port: 8083
endpoints:
- addresses:
- 10.10.0.7
- 10.10.0.8
conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamiudp3 name: whoamiudp3-abc
namespace: ns4 namespace: ns4
labels:
kubernetes.io/service-name: whoamiudp3
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.9
- ip: 10.10.0.10
ports:
- name: myapp4 - name: myapp4
port: 8084 port: 8084
endpoints:
- addresses:
- 10.10.0.9
- 10.10.0.10
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -118,18 +138,23 @@ spec:
task: whoamiudp-ipv6 task: whoamiudp-ipv6
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamiudp-ipv6 name: whoamiudp-ipv6-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamiudp-ipv6
subsets: addressType: IPv6
- addresses: ports:
- ip: "fd00:10:244:0:1::3"
ports:
- name: myapp-ipv6 - name: myapp-ipv6
port: 8080 port: 8080
endpoints:
- addresses:
- "fd00:10:244:0:1::3"
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -171,25 +196,30 @@ spec:
port: 80 port: 80
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamiudp-cross-ns name: whoamiudp-cross-ns-abc
namespace: cross-ns namespace: cross-ns
labels:
kubernetes.io/service-name: whoamiudp-cross-ns
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: myapp - name: myapp
port: 8000 port: 8000
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
kind: Service kind: Service
metadata: metadata:
name: whoamiudp-without-endpoints-subsets name: whoamiudp-without-endpointslice-endpoints
namespace: default namespace: default
spec: spec:
@ -202,11 +232,16 @@ spec:
task: whoamiudp task: whoamiudp
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamiudp-without-endpoints-subsets name: whoamiudp-without-endpointslice-endpoints-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamiudp-without-endpointslice-endpoints
addressType: IPv4
endpoints: []
--- ---
apiVersion: v1 apiVersion: v1

View file

@ -10,5 +10,5 @@ spec:
routes: routes:
- services: - services:
- name: whoamiudp-without-endpoints-subsets - name: whoamiudp-without-endpointslice-endpoints
port: 8000 port: 8000

View file

@ -0,0 +1,74 @@
apiVersion: v1
kind: Service
metadata:
name: whoami-svc-duplicated-endpointaddresses
namespace: default
spec:
ports:
- name: web
port: 8080
selector:
app: traefiklabs
task: whoami
---
kind: EndpointSlice
apiVersion: discovery.k8s.io/v1
metadata:
name: whoami-svc-duplicated-endpointaddresses-abc
namespace: default
labels:
kubernetes.io/service-name: whoami-svc-duplicated-endpointaddresses
addressType: IPv4
ports:
- name: web
port: 8080
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
---
kind: EndpointSlice
apiVersion: discovery.k8s.io/v1
metadata:
name: whoami-svc-duplicated-endpointaddresses-def
namespace: default
labels:
kubernetes.io/service-name: whoami-svc-duplicated-endpointaddresses
addressType: IPv4
ports:
- name: web
port: 8080
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
---
apiVersion: traefik.io/v1alpha1
kind: IngressRoute
metadata:
name: test.route
namespace: default
spec:
entryPoints:
- foo
routes:
- match: Host(`foo.com`) && PathPrefix(`/bar`)
kind: Rule
priority: 12
services:
- name: whoami-svc-duplicated-endpointaddresses
port: 8080

View file

@ -13,5 +13,5 @@ spec:
kind: Rule kind: Rule
priority: 12 priority: 12
services: services:
- name: whoami-without-endpoints-subsets - name: whoami-without-endpointslice-endpoints
port: 80 port: 80

View file

@ -30,7 +30,7 @@ metadata:
spec: spec:
weighted: weighted:
services: services:
- name: whoami-without-endpoints-subsets - name: whoami-without-endpointslice-endpoints
weight: 1 weight: 1
port: 80 port: 80
@ -43,10 +43,10 @@ metadata:
spec: spec:
mirroring: mirroring:
name: whoami-without-endpoints-subsets name: whoami-without-endpointslice-endpoints
port: 80 port: 80
mirrors: mirrors:
- name: whoami-without-endpoints-subsets - name: whoami-without-endpointslice-endpoints
port: 80 port: 80
- name: test-weighted - name: test-weighted
kind: TraefikService kind: TraefikService
@ -61,5 +61,5 @@ metadata:
spec: spec:
errors: errors:
service: service:
name: whoami-without-endpoints-subsets name: whoami-without-endpointslice-endpoints
port: 80 port: 80

View file

@ -1,17 +1,22 @@
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami4 name: whoami4-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami4
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -28,20 +33,25 @@ spec:
app: traefiklabs app: traefiklabs
task: whoami4 task: whoami4
------ ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami5 name: whoami5-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami5
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1

View file

@ -1,17 +1,22 @@
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami4 name: whoami4-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami4
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -28,20 +33,25 @@ spec:
app: traefiklabs app: traefiklabs
task: whoami4 task: whoami4
------ ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami5 name: whoami5-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami5
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1

View file

@ -0,0 +1,63 @@
apiVersion: v1
kind: Service
metadata:
name: whoami-svc-multiple-endpointaddresses
namespace: default
spec:
ports:
- name: web
port: 80
selector:
app: traefiklabs
task: whoami
---
kind: EndpointSlice
apiVersion: discovery.k8s.io/v1
metadata:
name: whoami-svc-multiple-endpointaddresses-abc
namespace: default
labels:
kubernetes.io/service-name: whoami-svc-multiple-endpointaddresses
addressType: IPv4
ports:
- name: web
port: 80
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: false
serving: true
- addresses:
- 10.10.0.5
- 10.10.0.6
conditions:
ready: true
---
apiVersion: traefik.io/v1alpha1
kind: IngressRoute
metadata:
name: test.route
namespace: default
spec:
entryPoints:
- foo
routes:
- match: Host(`foo.com`) && PathPrefix(`/bar`)
kind: Rule
priority: 12
services:
- name: whoami-svc-multiple-endpointaddresses
port: 80

View file

@ -0,0 +1,94 @@
apiVersion: v1
kind: Service
metadata:
name: whoami-svc-multiple-endpointslices
namespace: default
spec:
ports:
- name: web
port: 80
- name: web2
port: 8080
selector:
app: traefiklabs
task: whoami
---
kind: EndpointSlice
apiVersion: discovery.k8s.io/v1
metadata:
name: whoami-svc-multiple-endpointslices-abc
namespace: default
labels:
kubernetes.io/service-name: whoami-svc-multiple-endpointslices
addressType: IPv4
ports:
- name: web
port: 80
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
---
kind: EndpointSlice
apiVersion: discovery.k8s.io/v1
metadata:
name: whoami-svc-multiple-endpointslices-def
namespace: default
labels:
kubernetes.io/service-name: whoami-svc-multiple-endpointslices
addressType: IPv4
ports:
- name: web2
port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
---
kind: EndpointSlice
apiVersion: discovery.k8s.io/v1
metadata:
name: whoami-svc-multiple-endpointslices-ghi
namespace: default
labels:
kubernetes.io/service-name: whoami-svc-multiple-endpointslices
addressType: IPv4
ports:
- name: web2
port: 8080
endpoints:
- addresses:
- 10.10.0.5
- 10.10.0.6
conditions:
ready: true
---
apiVersion: traefik.io/v1alpha1
kind: IngressRoute
metadata:
name: test.route
namespace: default
spec:
entryPoints:
- foo
routes:
- match: Host(`foo.com`) && PathPrefix(`/bar`)
kind: Rule
priority: 12
services:
- name: whoami-svc-multiple-endpointslices
port: 8080

View file

@ -1,54 +0,0 @@
apiVersion: v1
kind: Service
metadata:
name: whoami-svc-multiple-subsets
namespace: default
spec:
ports:
- name: web
port: 80
- name: web2
port: 8080
selector:
app: traefiklabs
task: whoami
---
kind: Endpoints
apiVersion: v1
metadata:
name: whoami-svc-multiple-subsets
namespace: default
subsets:
- addresses:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: web
port: 80
- addresses:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: web2
port: 8080
---
apiVersion: traefik.io/v1alpha1
kind: IngressRoute
metadata:
name: test.route
namespace: default
spec:
entryPoints:
- foo
routes:
- match: Host(`foo.com`) && PathPrefix(`/bar`)
kind: Rule
priority: 12
services:
- name: whoami-svc-multiple-subsets
port: 8080

View file

@ -1,47 +1,62 @@
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami6 name: whoami6-abc
namespace: baz namespace: baz
labels:
kubernetes.io/service-name: whoami6
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.5
- ip: 10.10.0.6
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.5
- 10.10.0.6
conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami5 name: whoami5-abc
namespace: foo namespace: foo
labels:
kubernetes.io/service-name: whoami5
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami4 name: whoami4-abc
namespace: foo namespace: foo
labels:
kubernetes.io/service-name: whoami4
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1

View file

@ -1,17 +1,22 @@
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami5 name: whoami5-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami5
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1

View file

@ -1,62 +1,82 @@
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami4 name: whoami4-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami4
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: web - name: web
port: 80 port: 80
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami5 name: whoami5-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami5
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami6 name: whoami6-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami6
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.5
- ip: 10.10.0.6
ports:
- name: web - name: web
port: 80 port: 80
endpoints:
- addresses:
- 10.10.0.5
- 10.10.0.6
conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami7 name: whoami7-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami7
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.7
- ip: 10.10.0.8
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.7
- 10.10.0.8
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1

View file

@ -1,17 +1,22 @@
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami5 name: whoami5-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami5
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
apiVersion: traefik.io/v1alpha1 apiVersion: traefik.io/v1alpha1

View file

@ -1,32 +1,42 @@
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami5 name: whoami5-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami5
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami4 name: whoami4-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami4
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1

View file

@ -1,17 +1,22 @@
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami5 name: whoami5-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami5
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1

View file

@ -409,9 +409,8 @@ func (c configBuilder) loadServers(parentNamespace string, svc traefikv1alpha1.L
return nil, err return nil, err
} }
var servers []dynamic.Server
if service.Spec.Type != corev1.ServiceTypeExternalName && svc.HealthCheck != nil { if service.Spec.Type != corev1.ServiceTypeExternalName && svc.HealthCheck != nil {
return nil, fmt.Errorf("HealthCheck allowed only for ExternalName services: %s/%s", namespace, sanitizedName) return nil, fmt.Errorf("healthCheck allowed only for ExternalName services: %s/%s", namespace, sanitizedName)
} }
if service.Spec.Type == corev1.ServiceTypeExternalName { if service.Spec.Type == corev1.ServiceTypeExternalName {
@ -426,9 +425,7 @@ func (c configBuilder) loadServers(parentNamespace string, svc traefikv1alpha1.L
hostPort := net.JoinHostPort(service.Spec.ExternalName, strconv.Itoa(int(svcPort.Port))) hostPort := net.JoinHostPort(service.Spec.ExternalName, strconv.Itoa(int(svcPort.Port)))
return append(servers, dynamic.Server{ return []dynamic.Server{{URL: fmt.Sprintf("%s://%s", protocol, hostPort)}}, nil
URL: fmt.Sprintf("%s://%s", protocol, hostPort),
}), nil
} }
nativeLB := c.NativeLBByDefault nativeLB := c.NativeLBByDefault
@ -449,6 +446,7 @@ func (c configBuilder) loadServers(parentNamespace string, svc traefikv1alpha1.L
return []dynamic.Server{{URL: fmt.Sprintf("%s://%s", protocol, address)}}, nil return []dynamic.Server{{URL: fmt.Sprintf("%s://%s", protocol, address)}}, nil
} }
var servers []dynamic.Server
if service.Spec.Type == corev1.ServiceTypeNodePort && svc.NodePortLB { if service.Spec.Type == corev1.ServiceTypeNodePort && svc.NodePortLB {
nodes, nodesExists, nodesErr := c.client.GetNodes() nodes, nodesExists, nodesErr := c.client.GetNodes()
if nodesErr != nil { if nodesErr != nil {
@ -482,27 +480,20 @@ func (c configBuilder) loadServers(parentNamespace string, svc traefikv1alpha1.L
return servers, nil return servers, nil
} }
endpoints, endpointsExists, endpointsErr := c.client.GetEndpoints(namespace, sanitizedName) endpointSlices, err := c.client.GetEndpointSlicesForService(namespace, sanitizedName)
if endpointsErr != nil { if err != nil {
return nil, endpointsErr return nil, fmt.Errorf("getting endpointslices: %w", err)
}
if !endpointsExists {
return nil, fmt.Errorf("endpoints not found for %s/%s", namespace, sanitizedName)
} }
if len(endpoints.Subsets) == 0 && !c.allowEmptyServices { addresses := map[string]struct{}{}
return nil, fmt.Errorf("subset not found for %s/%s", namespace, sanitizedName) for _, endpointSlice := range endpointSlices {
}
for _, subset := range endpoints.Subsets {
var port int32 var port int32
for _, p := range subset.Ports { for _, p := range endpointSlice.Ports {
if svcPort.Name == p.Name { if svcPort.Name == *p.Name {
port = p.Port port = *p.Port
break break
} }
} }
if port == 0 { if port == 0 {
continue continue
} }
@ -512,14 +503,27 @@ func (c configBuilder) loadServers(parentNamespace string, svc traefikv1alpha1.L
return nil, err return nil, err
} }
for _, addr := range subset.Addresses { for _, endpoint := range endpointSlice.Endpoints {
hostPort := net.JoinHostPort(addr.IP, strconv.Itoa(int(port))) if endpoint.Conditions.Ready == nil || !*endpoint.Conditions.Ready {
continue
}
for _, address := range endpoint.Addresses {
if _, ok := addresses[address]; ok {
continue
}
addresses[address] = struct{}{}
servers = append(servers, dynamic.Server{ servers = append(servers, dynamic.Server{
URL: fmt.Sprintf("%s://%s", protocol, hostPort), URL: fmt.Sprintf("%s://%s", protocol, net.JoinHostPort(address, strconv.Itoa(int(port)))),
}) })
} }
} }
}
if len(servers) == 0 && !c.allowEmptyServices {
return nil, fmt.Errorf("no servers found for %s/%s", namespace, sanitizedName)
}
return servers, nil return servers, nil
} }

View file

@ -238,7 +238,6 @@ func (p *Provider) loadTCPServers(client Client, namespace string, svc traefikv1
} }
var servers []dynamic.TCPServer var servers []dynamic.TCPServer
if service.Spec.Type == corev1.ServiceTypeNodePort && svc.NodePortLB { if service.Spec.Type == corev1.ServiceTypeNodePort && svc.NodePortLB {
nodes, nodesExists, nodesErr := client.GetNodes() nodes, nodesExists, nodesErr := client.GetNodes()
if nodesErr != nil { if nodesErr != nil {
@ -284,39 +283,46 @@ func (p *Provider) loadTCPServers(client Client, namespace string, svc traefikv1
return []dynamic.TCPServer{{Address: address}}, nil return []dynamic.TCPServer{{Address: address}}, nil
} }
endpoints, endpointsExists, endpointsErr := client.GetEndpoints(namespace, svc.Name) endpointSlices, err := client.GetEndpointSlicesForService(namespace, svc.Name)
if endpointsErr != nil { if err != nil {
return nil, endpointsErr return nil, fmt.Errorf("getting endpointslices: %w", err)
}
if !endpointsExists {
return nil, errors.New("endpoints not found")
}
if len(endpoints.Subsets) == 0 && !p.AllowEmptyServices {
return nil, errors.New("subset not found")
} }
addresses := map[string]struct{}{}
for _, endpointSlice := range endpointSlices {
var port int32 var port int32
for _, subset := range endpoints.Subsets { for _, p := range endpointSlice.Ports {
for _, p := range subset.Ports { if svcPort.Name == *p.Name {
if svcPort.Name == p.Name { port = *p.Port
port = p.Port
break break
} }
} }
if port == 0 { if port == 0 {
return nil, errors.New("cannot define a port") continue
} }
for _, addr := range subset.Addresses { for _, endpoint := range endpointSlice.Endpoints {
if endpoint.Conditions.Ready == nil || !*endpoint.Conditions.Ready {
continue
}
for _, address := range endpoint.Addresses {
if _, ok := addresses[address]; ok {
continue
}
addresses[address] = struct{}{}
servers = append(servers, dynamic.TCPServer{ servers = append(servers, dynamic.TCPServer{
Address: net.JoinHostPort(addr.IP, strconv.Itoa(int(port))), Address: net.JoinHostPort(address, strconv.Itoa(int(port))),
}) })
} }
} }
} }
}
if len(servers) == 0 && !p.AllowEmptyServices {
return nil, fmt.Errorf("no servers found for %s/%s", namespace, svc.Name)
}
return servers, nil return servers, nil
} }

View file

@ -4620,9 +4620,9 @@ func TestLoadIngressRoutes(t *testing.T) {
}, },
}, },
{ {
desc: "IngressRoute, service with multiple subsets", desc: "IngressRoute, service with multiple endpoint addresses on endpointslice",
allowEmptyServices: true, allowEmptyServices: true,
paths: []string{"services.yml", "with_multiple_subsets.yml"}, paths: []string{"services.yml", "with_multiple_endpointaddresses.yml"},
expected: &dynamic.Configuration{ expected: &dynamic.Configuration{
UDP: &dynamic.UDPConfiguration{ UDP: &dynamic.UDPConfiguration{
Routers: map[string]*dynamic.UDPRouter{}, Routers: map[string]*dynamic.UDPRouter{},
@ -4648,6 +4648,66 @@ func TestLoadIngressRoutes(t *testing.T) {
"default-test-route-6b204d94623b3df4370c": { "default-test-route-6b204d94623b3df4370c": {
LoadBalancer: &dynamic.ServersLoadBalancer{ LoadBalancer: &dynamic.ServersLoadBalancer{
Servers: []dynamic.Server{ Servers: []dynamic.Server{
{
URL: "http://10.10.0.1:80",
},
{
URL: "http://10.10.0.2:80",
},
{
URL: "http://10.10.0.5:80",
},
{
URL: "http://10.10.0.6:80",
},
},
PassHostHeader: Bool(true),
ResponseForwarding: &dynamic.ResponseForwarding{
FlushInterval: ptypes.Duration(100 * time.Millisecond),
},
},
},
},
ServersTransports: map[string]*dynamic.ServersTransport{},
},
TLS: &dynamic.TLSConfiguration{},
},
},
{
desc: "IngressRoute, service with duplicated endpointaddresses",
allowEmptyServices: true,
paths: []string{"services.yml", "with_duplicated_endpointaddresses.yml"},
expected: &dynamic.Configuration{
UDP: &dynamic.UDPConfiguration{
Routers: map[string]*dynamic.UDPRouter{},
Services: map[string]*dynamic.UDPService{},
},
TCP: &dynamic.TCPConfiguration{
Routers: map[string]*dynamic.TCPRouter{},
Middlewares: map[string]*dynamic.TCPMiddleware{},
Services: map[string]*dynamic.TCPService{},
ServersTransports: map[string]*dynamic.TCPServersTransport{},
},
HTTP: &dynamic.HTTPConfiguration{
Routers: map[string]*dynamic.Router{
"default-test-route-6b204d94623b3df4370c": {
EntryPoints: []string{"foo"},
Service: "default-test-route-6b204d94623b3df4370c",
Rule: "Host(`foo.com`) && PathPrefix(`/bar`)",
Priority: 12,
},
},
Middlewares: map[string]*dynamic.Middleware{},
Services: map[string]*dynamic.Service{
"default-test-route-6b204d94623b3df4370c": {
LoadBalancer: &dynamic.ServersLoadBalancer{
Servers: []dynamic.Server{
{
URL: "http://10.10.0.1:8080",
},
{
URL: "http://10.10.0.2:8080",
},
{ {
URL: "http://10.10.0.3:8080", URL: "http://10.10.0.3:8080",
}, },
@ -4726,7 +4786,7 @@ func TestLoadIngressRoutes(t *testing.T) {
Weighted: &dynamic.WeightedRoundRobin{ Weighted: &dynamic.WeightedRoundRobin{
Services: []dynamic.WRRService{ Services: []dynamic.WRRService{
{ {
Name: "default-whoami-without-endpoints-subsets-80", Name: "default-whoami-without-endpointslice-endpoints-80",
Weight: func(i int) *int { return &i }(1), Weight: func(i int) *int { return &i }(1),
}, },
}, },
@ -4734,10 +4794,10 @@ func TestLoadIngressRoutes(t *testing.T) {
}, },
"default-test-mirror": { "default-test-mirror": {
Mirroring: &dynamic.Mirroring{ Mirroring: &dynamic.Mirroring{
Service: "default-whoami-without-endpoints-subsets-80", Service: "default-whoami-without-endpointslice-endpoints-80",
Mirrors: []dynamic.MirrorService{ Mirrors: []dynamic.MirrorService{
{ {
Name: "default-whoami-without-endpoints-subsets-80", Name: "default-whoami-without-endpointslice-endpoints-80",
}, },
{ {
Name: "default-test-weighted", Name: "default-test-weighted",
@ -4745,7 +4805,7 @@ func TestLoadIngressRoutes(t *testing.T) {
}, },
}, },
}, },
"default-whoami-without-endpoints-subsets-80": { "default-whoami-without-endpointslice-endpoints-80": {
LoadBalancer: &dynamic.ServersLoadBalancer{ LoadBalancer: &dynamic.ServersLoadBalancer{
PassHostHeader: Bool(true), PassHostHeader: Bool(true),
ResponseForwarding: &dynamic.ResponseForwarding{ ResponseForwarding: &dynamic.ResponseForwarding{
@ -4799,6 +4859,87 @@ func TestLoadIngressRoutes(t *testing.T) {
} }
} }
func TestLoadIngressRoutes_multipleEndpointAddresses(t *testing.T) {
wantConf := &dynamic.Configuration{
UDP: &dynamic.UDPConfiguration{
Routers: map[string]*dynamic.UDPRouter{},
Services: map[string]*dynamic.UDPService{},
},
TCP: &dynamic.TCPConfiguration{
Routers: map[string]*dynamic.TCPRouter{},
Middlewares: map[string]*dynamic.TCPMiddleware{},
Services: map[string]*dynamic.TCPService{},
ServersTransports: map[string]*dynamic.TCPServersTransport{},
},
HTTP: &dynamic.HTTPConfiguration{
Routers: map[string]*dynamic.Router{
"default-test-route-6b204d94623b3df4370c": {
EntryPoints: []string{"foo"},
Service: "default-test-route-6b204d94623b3df4370c",
Rule: "Host(`foo.com`) && PathPrefix(`/bar`)",
Priority: 12,
},
},
Middlewares: map[string]*dynamic.Middleware{},
Services: map[string]*dynamic.Service{
"default-test-route-6b204d94623b3df4370c": {
LoadBalancer: &dynamic.ServersLoadBalancer{
PassHostHeader: Bool(true),
ResponseForwarding: &dynamic.ResponseForwarding{
FlushInterval: ptypes.Duration(100 * time.Millisecond),
},
},
},
},
ServersTransports: map[string]*dynamic.ServersTransport{},
},
TLS: &dynamic.TLSConfiguration{},
}
wantServers := []dynamic.Server{
{
URL: "http://10.10.0.3:8080",
},
{
URL: "http://10.10.0.4:8080",
},
{
URL: "http://10.10.0.5:8080",
},
{
URL: "http://10.10.0.6:8080",
},
}
k8sObjects, crdObjects := readResources(t, []string{"services.yml", "with_multiple_endpointslices.yml"})
kubeClient := kubefake.NewSimpleClientset(k8sObjects...)
crdClient := traefikcrdfake.NewSimpleClientset(crdObjects...)
client := newClientImpl(kubeClient, crdClient)
stopCh := make(chan struct{})
eventCh, err := client.WatchAll(nil, stopCh)
require.NoError(t, err)
if k8sObjects != nil || crdObjects != nil {
// just wait for the first event
<-eventCh
}
p := Provider{}
conf := p.loadConfigurationFromCRD(context.Background(), client)
service, ok := conf.HTTP.Services["default-test-route-6b204d94623b3df4370c"]
require.True(t, ok)
require.NotNil(t, service)
require.NotNil(t, service.LoadBalancer)
assert.ElementsMatch(t, wantServers, service.LoadBalancer.Servers)
service.LoadBalancer.Servers = nil
assert.Equal(t, wantConf, conf)
}
func TestLoadIngressRouteUDPs(t *testing.T) { func TestLoadIngressRouteUDPs(t *testing.T) {
testCases := []struct { testCases := []struct {
desc string desc string

View file

@ -122,7 +122,6 @@ func (p *Provider) loadUDPServers(client Client, namespace string, svc traefikv1
} }
var servers []dynamic.UDPServer var servers []dynamic.UDPServer
if service.Spec.Type == corev1.ServiceTypeNodePort && svc.NodePortLB { if service.Spec.Type == corev1.ServiceTypeNodePort && svc.NodePortLB {
nodes, nodesExists, nodesErr := client.GetNodes() nodes, nodesExists, nodesErr := client.GetNodes()
if nodesErr != nil { if nodesErr != nil {
@ -168,39 +167,46 @@ func (p *Provider) loadUDPServers(client Client, namespace string, svc traefikv1
return []dynamic.UDPServer{{Address: address}}, nil return []dynamic.UDPServer{{Address: address}}, nil
} }
endpoints, endpointsExists, endpointsErr := client.GetEndpoints(namespace, svc.Name) endpointSlices, err := client.GetEndpointSlicesForService(namespace, svc.Name)
if endpointsErr != nil { if err != nil {
return nil, endpointsErr return nil, fmt.Errorf("getting endpointslices: %w", err)
}
if !endpointsExists {
return nil, errors.New("endpoints not found")
}
if len(endpoints.Subsets) == 0 && !p.AllowEmptyServices {
return nil, errors.New("subset not found")
} }
addresses := map[string]struct{}{}
for _, endpointSlice := range endpointSlices {
var port int32 var port int32
for _, subset := range endpoints.Subsets { for _, p := range endpointSlice.Ports {
for _, p := range subset.Ports { if svcPort.Name == *p.Name {
if svcPort.Name == p.Name { port = *p.Port
port = p.Port
break break
} }
} }
if port == 0 { if port == 0 {
return nil, errors.New("cannot define a port") continue
} }
for _, addr := range subset.Addresses { for _, endpoint := range endpointSlice.Endpoints {
if endpoint.Conditions.Ready == nil || !*endpoint.Conditions.Ready {
continue
}
for _, address := range endpoint.Addresses {
if _, ok := addresses[address]; ok {
continue
}
addresses[address] = struct{}{}
servers = append(servers, dynamic.UDPServer{ servers = append(servers, dynamic.UDPServer{
Address: net.JoinHostPort(addr.IP, strconv.Itoa(int(port))), Address: net.JoinHostPort(address, strconv.Itoa(int(port))),
}) })
} }
} }
} }
}
if len(servers) == 0 && !p.AllowEmptyServices {
return nil, fmt.Errorf("no servers found for %s/%s", namespace, svc.Name)
}
return servers, nil return servers, nil
} }

View file

@ -11,9 +11,11 @@ import (
"github.com/rs/zerolog/log" "github.com/rs/zerolog/log"
"github.com/traefik/traefik/v3/pkg/types" "github.com/traefik/traefik/v3/pkg/types"
corev1 "k8s.io/api/core/v1" corev1 "k8s.io/api/core/v1"
discoveryv1 "k8s.io/api/discovery/v1"
kerror "k8s.io/apimachinery/pkg/api/errors" kerror "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/labels" "k8s.io/apimachinery/pkg/labels"
"k8s.io/apimachinery/pkg/selection"
ktypes "k8s.io/apimachinery/pkg/types" ktypes "k8s.io/apimachinery/pkg/types"
kinformers "k8s.io/client-go/informers" kinformers "k8s.io/client-go/informers"
kclientset "k8s.io/client-go/kubernetes" kclientset "k8s.io/client-go/kubernetes"
@ -61,9 +63,9 @@ type Client interface {
ListTLSRoutes() ([]*gatev1alpha2.TLSRoute, error) ListTLSRoutes() ([]*gatev1alpha2.TLSRoute, error)
ListNamespaces(selector labels.Selector) ([]string, error) ListNamespaces(selector labels.Selector) ([]string, error)
ListReferenceGrants(namespace string) ([]*gatev1beta1.ReferenceGrant, error) ListReferenceGrants(namespace string) ([]*gatev1beta1.ReferenceGrant, error)
ListEndpointSlicesForService(namespace, serviceName string) ([]*discoveryv1.EndpointSlice, error)
GetService(namespace, name string) (*corev1.Service, bool, error) GetService(namespace, name string) (*corev1.Service, bool, error)
GetSecret(namespace, name string) (*corev1.Secret, bool, error) GetSecret(namespace, name string) (*corev1.Secret, bool, error)
GetEndpoints(namespace, name string) (*corev1.Endpoints, bool, error)
} }
type clientWrapper struct { type clientWrapper struct {
@ -222,7 +224,7 @@ func (c *clientWrapper) WatchAll(namespaces []string, stopCh <-chan struct{}) (<
if err != nil { if err != nil {
return nil, err return nil, err
} }
_, err = factoryKube.Core().V1().Endpoints().Informer().AddEventHandler(eventHandler) _, err = factoryKube.Discovery().V1().EndpointSlices().Informer().AddEventHandler(eventHandler)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -543,16 +545,20 @@ func (c *clientWrapper) GetService(namespace, name string) (*corev1.Service, boo
return service, exist, err return service, exist, err
} }
// GetEndpoints returns the named endpoints from the given namespace. // ListEndpointSlicesForService returns the EndpointSlices for the given service name in the given namespace.
func (c *clientWrapper) GetEndpoints(namespace, name string) (*corev1.Endpoints, bool, error) { func (c *clientWrapper) ListEndpointSlicesForService(namespace, serviceName string) ([]*discoveryv1.EndpointSlice, error) {
if !c.isWatchedNamespace(namespace) { if !c.isWatchedNamespace(namespace) {
return nil, false, fmt.Errorf("failed to get endpoints %s/%s: namespace is not within watched namespaces", namespace, name) return nil, fmt.Errorf("failed to get endpointslices for service %s/%s: namespace is not within watched namespaces", namespace, serviceName)
} }
endpoint, err := c.factoriesKube[c.lookupNamespace(namespace)].Core().V1().Endpoints().Lister().Endpoints(namespace).Get(name) serviceLabelRequirement, err := labels.NewRequirement(discoveryv1.LabelServiceName, selection.Equals, []string{serviceName})
exist, err := translateNotFoundError(err) if err != nil {
return nil, fmt.Errorf("failed to create service label selector requirement: %w", err)
}
serviceSelector := labels.NewSelector()
serviceSelector = serviceSelector.Add(*serviceLabelRequirement)
return endpoint, exist, err return c.factoriesKube[c.lookupNamespace(namespace)].Discovery().V1().EndpointSlices().Lister().EndpointSlices(namespace).List(serviceSelector)
} }
// GetSecret returns the named secret from the given namespace. // GetSecret returns the named secret from the given namespace.

View file

@ -0,0 +1,50 @@
---
kind: GatewayClass
apiVersion: gateway.networking.k8s.io/v1
metadata:
name: my-gateway-class
spec:
controllerName: traefik.io/gateway-controller
---
kind: Gateway
apiVersion: gateway.networking.k8s.io/v1
metadata:
name: my-gateway
namespace: default
spec:
gatewayClassName: my-gateway-class
listeners: # Use GatewayClass defaults for listener definition.
- name: http
protocol: HTTP
port: 80
allowedRoutes:
namespaces:
from: Same
---
kind: HTTPRoute
apiVersion: gateway.networking.k8s.io/v1
metadata:
name: http-app-1
namespace: default
spec:
parentRefs:
- name: my-gateway
kind: Gateway
group: gateway.networking.k8s.io
hostnames:
- "foo.com"
rules:
- matches:
- method: GET
path:
type: PathPrefix
value: /foo
backendRefs:
- name: whoami
port: 80
weight: 1
kind: Service
group: ""

View file

@ -0,0 +1,56 @@
---
kind: GatewayClass
apiVersion: gateway.networking.k8s.io/v1
metadata:
name: my-gateway-class
spec:
controllerName: traefik.io/gateway-controller
---
kind: Gateway
apiVersion: gateway.networking.k8s.io/v1
metadata:
name: my-gateway
namespace: default
spec:
gatewayClassName: my-gateway-class
listeners: # Use GatewayClass defaults for listener definition.
- name: http
protocol: HTTP
port: 80
allowedRoutes:
namespaces:
from: Same
---
kind: HTTPRoute
apiVersion: gateway.networking.k8s.io/v1
metadata:
name: http-app-1
namespace: default
spec:
parentRefs:
- name: my-gateway
kind: Gateway
group: gateway.networking.k8s.io
hostnames:
- "foo.com"
rules:
- matches:
- queryParams:
- type: Exact
name: foo
value: bar
- type: RegularExpression
name: baz
value: buz
path:
type: PathPrefix
value: /foo
backendRefs:
- name: whoami
port: 80
weight: 1
kind: Service
group: ""

View file

@ -17,21 +17,26 @@ spec:
task: whoami task: whoami
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami name: whoami-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: web - name: web
port: 80 port: 80
- name: web2 - name: web2
port: 8000 port: 8080
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -53,21 +58,26 @@ spec:
task: whoami task: whoami
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami-bar name: whoami-bar-abc
namespace: bar namespace: bar
labels:
kubernetes.io/service-name: whoami-bar
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.11
- ip: 10.10.0.12
ports:
- name: web - name: web
port: 80 port: 80
- name: web2 - name: web2
port: 8000 port: 8000
endpoints:
- addresses:
- 10.10.0.11
- 10.10.0.12
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -86,19 +96,24 @@ spec:
task: whoami2 task: whoami2
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami2 name: whoami2-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami2
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.3
- ip: 10.10.0.4
ports:
- name: web - name: web
port: 8080 port: 8080
endpoints:
- addresses:
- 10.10.0.3
- 10.10.0.4
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -117,19 +132,24 @@ spec:
task: whoami2 task: whoami2
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitls name: whoamitls-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamitls
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.5
- ip: 10.10.0.6
ports:
- name: websecure - name: websecure
port: 8443 port: 8443
endpoints:
- addresses:
- 10.10.0.5
- 10.10.0.6
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -148,19 +168,24 @@ spec:
task: whoami3 task: whoami3
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoami3 name: whoami3-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoami3
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.7
- ip: 10.10.0.8
ports:
- name: websecure2 - name: websecure2
port: 8443 port: 8443
endpoints:
- addresses:
- 10.10.0.7
- 10.10.0.8
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -201,23 +226,28 @@ spec:
port: 443 port: 443
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitcp name: whoamitcp-abc
namespace: default namespace: default
labels:
kubernetes.io/service-name: whoamitcp
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.9
- ip: 10.10.0.10
ports:
- name: tcp-1 - name: tcp-1
protocol: TCP protocol: TCP
port: 9000 port: 9000
- name: tcp-2 - name: tcp-2
protocol: TCP protocol: TCP
port: 10000 port: 10000
endpoints:
- addresses:
- 10.10.0.9
- 10.10.0.10
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1
@ -236,23 +266,28 @@ spec:
name: tcp-2 name: tcp-2
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: whoamitcp-bar name: whoamitcp-bar-abc
namespace: bar namespace: bar
labels:
kubernetes.io/service-name: whoamitcp-bar
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.13
- ip: 10.10.0.14
ports:
- name: tcp-1 - name: tcp-1
protocol: TCP protocol: TCP
port: 9000 port: 9000
- name: tcp-2 - name: tcp-2
protocol: TCP protocol: TCP
port: 10000 port: 10000
endpoints:
- addresses:
- 10.10.0.13
- 10.10.0.14
conditions:
ready: true
--- ---
apiVersion: v1 apiVersion: v1

View file

@ -370,6 +370,10 @@ func (p *Provider) loadHTTPRouteFilterExtensionRef(namespace string, extensionRe
} }
func (p *Provider) loadHTTPServers(namespace string, backendRef gatev1.HTTPBackendRef) (*dynamic.ServersLoadBalancer, error) { func (p *Provider) loadHTTPServers(namespace string, backendRef gatev1.HTTPBackendRef) (*dynamic.ServersLoadBalancer, error) {
if backendRef.Port == nil {
return nil, errors.New("port is required for Kubernetes Service reference")
}
service, exists, err := p.client.GetService(namespace, string(backendRef.Name)) service, exists, err := p.client.GetService(namespace, string(backendRef.Name))
if err != nil { if err != nil {
return nil, fmt.Errorf("getting service: %w", err) return nil, fmt.Errorf("getting service: %w", err)
@ -378,58 +382,60 @@ func (p *Provider) loadHTTPServers(namespace string, backendRef gatev1.HTTPBacke
return nil, errors.New("service not found") return nil, errors.New("service not found")
} }
var portSpec corev1.ServicePort var svcPort *corev1.ServicePort
var match bool
for _, p := range service.Spec.Ports { for _, p := range service.Spec.Ports {
if backendRef.Port == nil || p.Port == int32(*backendRef.Port) { if p.Port == int32(*backendRef.Port) {
portSpec = p svcPort = &p
match = true
break break
} }
} }
if !match { if svcPort == nil {
return nil, errors.New("service port not found") return nil, fmt.Errorf("service port %d not found", *backendRef.Port)
} }
endpoints, endpointsExists, err := p.client.GetEndpoints(namespace, string(backendRef.Name)) endpointSlices, err := p.client.ListEndpointSlicesForService(namespace, string(backendRef.Name))
if err != nil { if err != nil {
return nil, fmt.Errorf("getting endpoints: %w", err) return nil, fmt.Errorf("getting endpointslices: %w", err)
} }
if !endpointsExists { if len(endpointSlices) == 0 {
return nil, errors.New("endpoints not found") return nil, errors.New("endpointslices not found")
}
if len(endpoints.Subsets) == 0 {
return nil, errors.New("subset not found")
} }
lb := &dynamic.ServersLoadBalancer{} lb := &dynamic.ServersLoadBalancer{}
lb.SetDefaults() lb.SetDefaults()
protocol := getProtocol(*svcPort)
addresses := map[string]struct{}{}
for _, endpointSlice := range endpointSlices {
var port int32 var port int32
var portStr string for _, p := range endpointSlice.Ports {
for _, subset := range endpoints.Subsets { if svcPort.Name == *p.Name {
for _, p := range subset.Ports { port = *p.Port
if portSpec.Name == p.Name {
port = p.Port
break break
} }
} }
if port == 0 { if port == 0 {
return nil, errors.New("cannot define a port") continue
} }
protocol := getProtocol(portSpec) for _, endpoint := range endpointSlice.Endpoints {
if endpoint.Conditions.Ready == nil || !*endpoint.Conditions.Ready {
continue
}
portStr = strconv.FormatInt(int64(port), 10) for _, address := range endpoint.Addresses {
for _, addr := range subset.Addresses { if _, ok := addresses[address]; ok {
continue
}
addresses[address] = struct{}{}
lb.Servers = append(lb.Servers, dynamic.Server{ lb.Servers = append(lb.Servers, dynamic.Server{
URL: fmt.Sprintf("%s://%s", protocol, net.JoinHostPort(addr.IP, portStr)), URL: fmt.Sprintf("%s://%s", protocol, net.JoinHostPort(address, strconv.Itoa(int(port)))),
}) })
} }
} }
}
return lb, nil return lb, nil
} }
@ -469,11 +475,11 @@ func buildHostRule(hostnames []gatev1.Hostname) (string, int) {
// The current priority computing is rather naive but aims to fulfill Conformance tests suite requirement. // The current priority computing is rather naive but aims to fulfill Conformance tests suite requirement.
// The priority is computed to match the following precedence order: // The priority is computed to match the following precedence order:
// //
// * "Exact" path match. (+100000) // * "Exact" path match (+100000).
// * "Prefix" path match with largest number of characters. (+10000) PathRegex (+1000) // * "Prefix" path match with largest number of characters (+10000 + nb_characters*100).
// * Method match. (not implemented) // * Method match (+1000).
// * Largest number of header matches. (+100 each) or with PathRegex (+10 each) // * Largest number of header matches (+100 each).
// * Largest number of query param matches. (not implemented) // * Largest number of query param matches (+10 each).
// //
// In case of multiple matches for a route, the maximum priority among all matches is retain. // In case of multiple matches for a route, the maximum priority among all matches is retain.
func buildMatchRule(hostnames []gatev1.Hostname, match gatev1.HTTPRouteMatch) (string, int) { func buildMatchRule(hostnames []gatev1.Hostname, match gatev1.HTTPRouteMatch) (string, int) {
@ -489,10 +495,19 @@ func buildMatchRule(hostnames []gatev1.Hostname, match gatev1.HTTPRouteMatch) (s
matchRules = append(matchRules, pathRule) matchRules = append(matchRules, pathRule)
priority += pathPriority priority += pathPriority
if match.Method != nil {
matchRules = append(matchRules, fmt.Sprintf("Method(`%s`)", *match.Method))
priority += 1000
}
headerRules, headersPriority := buildHeaderRules(match.Headers) headerRules, headersPriority := buildHeaderRules(match.Headers)
matchRules = append(matchRules, headerRules...) matchRules = append(matchRules, headerRules...)
priority += headersPriority priority += headersPriority
queryParamRules, queryParamsPriority := buildQueryParamRules(match.QueryParams)
matchRules = append(matchRules, queryParamRules...)
priority += queryParamsPriority
matchRulesStr := strings.Join(matchRules, " && ") matchRulesStr := strings.Join(matchRules, " && ")
hostRule, hostPriority := buildHostRule(hostnames) hostRule, hostPriority := buildHostRule(hostnames)
@ -525,7 +540,7 @@ func buildPathRule(pathMatch gatev1.HTTPPathMatch) (string, int) {
return fmt.Sprintf("(Path(`%[1]s`) || PathPrefix(`%[1]s/`))", pv), 10000 + len(pathValue)*100 return fmt.Sprintf("(Path(`%[1]s`) || PathPrefix(`%[1]s/`))", pv), 10000 + len(pathValue)*100
case gatev1.PathMatchRegularExpression: case gatev1.PathMatchRegularExpression:
return fmt.Sprintf("PathRegexp(`%s`)", pathValue), 1000 + len(pathValue)*100 return fmt.Sprintf("PathRegexp(`%s`)", pathValue), 10000 + len(pathValue)*100
default: default:
return "PathPrefix(`/`)", 1 return "PathPrefix(`/`)", 1
@ -533,18 +548,38 @@ func buildPathRule(pathMatch gatev1.HTTPPathMatch) (string, int) {
} }
func buildHeaderRules(headers []gatev1.HTTPHeaderMatch) ([]string, int) { func buildHeaderRules(headers []gatev1.HTTPHeaderMatch) ([]string, int) {
var rules []string var (
var priority int rules []string
priority int
)
for _, header := range headers { for _, header := range headers {
typ := ptr.Deref(header.Type, gatev1.HeaderMatchExact) typ := ptr.Deref(header.Type, gatev1.HeaderMatchExact)
switch typ { switch typ {
case gatev1.HeaderMatchExact: case gatev1.HeaderMatchExact:
rules = append(rules, fmt.Sprintf("Header(`%s`,`%s`)", header.Name, header.Value)) rules = append(rules, fmt.Sprintf("Header(`%s`,`%s`)", header.Name, header.Value))
priority += 100
case gatev1.HeaderMatchRegularExpression: case gatev1.HeaderMatchRegularExpression:
rules = append(rules, fmt.Sprintf("HeaderRegexp(`%s`,`%s`)", header.Name, header.Value)) rules = append(rules, fmt.Sprintf("HeaderRegexp(`%s`,`%s`)", header.Name, header.Value))
priority += 10
} }
priority += 100
}
return rules, priority
}
func buildQueryParamRules(queryParams []gatev1.HTTPQueryParamMatch) ([]string, int) {
var (
rules []string
priority int
)
for _, qp := range queryParams {
typ := ptr.Deref(qp.Type, gatev1.QueryParamMatchExact)
switch typ {
case gatev1.QueryParamMatchExact:
rules = append(rules, fmt.Sprintf("Query(`%s`,`%s`)", qp.Name, qp.Value))
case gatev1.QueryParamMatchRegularExpression:
rules = append(rules, fmt.Sprintf("QueryRegexp(`%s`,`%s`)", qp.Name, qp.Value))
}
priority += 10
} }
return rules, priority return rules, priority

View file

@ -1288,7 +1288,7 @@ func TestLoadHTTPRoutes(t *testing.T) {
"default-http-app-1-my-gateway-web-2-d23f7039bc8036fb918c": { "default-http-app-1-my-gateway-web-2-d23f7039bc8036fb918c": {
EntryPoints: []string{"web"}, EntryPoints: []string{"web"},
Rule: "Host(`foo.com`) && PathRegexp(`^/buzz/[0-9]+$`)", Rule: "Host(`foo.com`) && PathRegexp(`^/buzz/[0-9]+$`)",
Priority: 2408, Priority: 11408,
RuleSyntax: "v3", RuleSyntax: "v3",
Service: "default-http-app-1-my-gateway-web-2-wrr", Service: "default-http-app-1-my-gateway-web-2-wrr",
}, },
@ -1354,6 +1354,128 @@ func TestLoadHTTPRoutes(t *testing.T) {
TLS: &dynamic.TLSConfiguration{}, TLS: &dynamic.TLSConfiguration{},
}, },
}, },
{
desc: "Simple HTTPRoute, with method matching",
paths: []string{"services.yml", "httproute/with_method_matching.yml"},
entryPoints: map[string]Entrypoint{"web": {
Address: ":80",
}},
expected: &dynamic.Configuration{
UDP: &dynamic.UDPConfiguration{
Routers: map[string]*dynamic.UDPRouter{},
Services: map[string]*dynamic.UDPService{},
},
TCP: &dynamic.TCPConfiguration{
Routers: map[string]*dynamic.TCPRouter{},
Middlewares: map[string]*dynamic.TCPMiddleware{},
Services: map[string]*dynamic.TCPService{},
ServersTransports: map[string]*dynamic.TCPServersTransport{},
},
HTTP: &dynamic.HTTPConfiguration{
Routers: map[string]*dynamic.Router{
"default-http-app-1-my-gateway-web-0-74ad70a7cf090becdd3c": {
EntryPoints: []string{"web"},
Rule: "Host(`foo.com`) && (Path(`/foo`) || PathPrefix(`/foo/`)) && Method(`GET`)",
Priority: 11408,
RuleSyntax: "v3",
Service: "default-http-app-1-my-gateway-web-0-wrr",
},
},
Middlewares: map[string]*dynamic.Middleware{},
Services: map[string]*dynamic.Service{
"default-http-app-1-my-gateway-web-0-wrr": {
Weighted: &dynamic.WeightedRoundRobin{
Services: []dynamic.WRRService{
{
Name: "default-whoami-80",
Weight: ptr.To(1),
},
},
},
},
"default-whoami-80": {
LoadBalancer: &dynamic.ServersLoadBalancer{
Servers: []dynamic.Server{
{
URL: "http://10.10.0.1:80",
},
{
URL: "http://10.10.0.2:80",
},
},
PassHostHeader: ptr.To(true),
ResponseForwarding: &dynamic.ResponseForwarding{
FlushInterval: ptypes.Duration(100 * time.Millisecond),
},
},
},
},
ServersTransports: map[string]*dynamic.ServersTransport{},
},
TLS: &dynamic.TLSConfiguration{},
},
},
{
desc: "Simple HTTPRoute, with query param matching",
paths: []string{"services.yml", "httproute/with_query_param_matching.yml"},
entryPoints: map[string]Entrypoint{"web": {
Address: ":80",
}},
expected: &dynamic.Configuration{
UDP: &dynamic.UDPConfiguration{
Routers: map[string]*dynamic.UDPRouter{},
Services: map[string]*dynamic.UDPService{},
},
TCP: &dynamic.TCPConfiguration{
Routers: map[string]*dynamic.TCPRouter{},
Middlewares: map[string]*dynamic.TCPMiddleware{},
Services: map[string]*dynamic.TCPService{},
ServersTransports: map[string]*dynamic.TCPServersTransport{},
},
HTTP: &dynamic.HTTPConfiguration{
Routers: map[string]*dynamic.Router{
"default-http-app-1-my-gateway-web-0-bb7b03c9610e982fd627": {
EntryPoints: []string{"web"},
Rule: "Host(`foo.com`) && (Path(`/foo`) || PathPrefix(`/foo/`)) && Query(`foo`,`bar`) && QueryRegexp(`baz`,`buz`)",
Priority: 10428,
RuleSyntax: "v3",
Service: "default-http-app-1-my-gateway-web-0-wrr",
},
},
Middlewares: map[string]*dynamic.Middleware{},
Services: map[string]*dynamic.Service{
"default-http-app-1-my-gateway-web-0-wrr": {
Weighted: &dynamic.WeightedRoundRobin{
Services: []dynamic.WRRService{
{
Name: "default-whoami-80",
Weight: ptr.To(1),
},
},
},
},
"default-whoami-80": {
LoadBalancer: &dynamic.ServersLoadBalancer{
Servers: []dynamic.Server{
{
URL: "http://10.10.0.1:80",
},
{
URL: "http://10.10.0.2:80",
},
},
PassHostHeader: ptr.To(true),
ResponseForwarding: &dynamic.ResponseForwarding{
FlushInterval: ptypes.Duration(100 * time.Millisecond),
},
},
},
},
ServersTransports: map[string]*dynamic.ServersTransport{},
},
TLS: &dynamic.TLSConfiguration{},
},
},
{ {
desc: "HTTPRoute with Same namespace selector", desc: "HTTPRoute with Same namespace selector",
paths: []string{"services.yml", "httproute/with_namespace_same.yml"}, paths: []string{"services.yml", "httproute/with_namespace_same.yml"},

View file

@ -206,82 +206,71 @@ func (p *Provider) loadTCPServices(namespace string, backendRefs []gatev1.Backen
return nil, nil, fmt.Errorf("unsupported BackendRef %s/%s/%s", *backendRef.Group, *backendRef.Kind, backendRef.Name) return nil, nil, fmt.Errorf("unsupported BackendRef %s/%s/%s", *backendRef.Group, *backendRef.Kind, backendRef.Name)
} }
svc := dynamic.TCPService{ if backendRef.Port == nil {
LoadBalancer: &dynamic.TCPServersLoadBalancer{}, return nil, nil, errors.New("port is required for Kubernetes Service reference")
} }
service, exists, err := p.client.GetService(namespace, string(backendRef.Name)) service, exists, err := p.client.GetService(namespace, string(backendRef.Name))
if err != nil { if err != nil {
return nil, nil, err return nil, nil, fmt.Errorf("getting service: %w", err)
} }
if !exists { if !exists {
return nil, nil, errors.New("service not found") return nil, nil, errors.New("service not found")
} }
if len(service.Spec.Ports) > 1 && backendRef.Port == nil { var svcPort *corev1.ServicePort
// If the port is unspecified and the backend is a Service for _, p := range service.Spec.Ports {
// object consisting of multiple port definitions, the route if p.Port == int32(*backendRef.Port) {
// must be dropped from the Gateway. The controller should svcPort = &p
// raise the "ResolvedRefs" condition on the Gateway with the break
// "DroppedRoutes" reason. The gateway status for this route }
// should be updated with a condition that describes the error }
// more specifically. if svcPort == nil {
log.Error().Msg("A multiple ports Kubernetes Service cannot be used if unspecified backendRef.Port") return nil, nil, fmt.Errorf("service port %d not found", *backendRef.Port)
}
endpointSlices, err := p.client.ListEndpointSlicesForService(namespace, string(backendRef.Name))
if err != nil {
return nil, nil, fmt.Errorf("getting endpointslices: %w", err)
}
if len(endpointSlices) == 0 {
return nil, nil, errors.New("endpointslices not found")
}
svc := dynamic.TCPService{LoadBalancer: &dynamic.TCPServersLoadBalancer{}}
addresses := map[string]struct{}{}
for _, endpointSlice := range endpointSlices {
var port int32
for _, p := range endpointSlice.Ports {
if svcPort.Name == *p.Name {
port = *p.Port
break
}
}
if port == 0 {
continue continue
} }
var portSpec corev1.ServicePort for _, endpoint := range endpointSlice.Endpoints {
var match bool if endpoint.Conditions.Ready == nil || !*endpoint.Conditions.Ready {
continue
for _, p := range service.Spec.Ports {
if backendRef.Port == nil || p.Port == int32(*backendRef.Port) {
portSpec = p
match = true
break
}
} }
if !match { for _, address := range endpoint.Addresses {
return nil, nil, errors.New("service port not found") if _, ok := addresses[address]; ok {
continue
} }
endpoints, endpointsExists, endpointsErr := p.client.GetEndpoints(namespace, string(backendRef.Name)) addresses[address] = struct{}{}
if endpointsErr != nil {
return nil, nil, endpointsErr
}
if !endpointsExists {
return nil, nil, errors.New("endpoints not found")
}
if len(endpoints.Subsets) == 0 {
return nil, nil, errors.New("subset not found")
}
var port int32
var portStr string
for _, subset := range endpoints.Subsets {
for _, p := range subset.Ports {
if portSpec.Name == p.Name {
port = p.Port
break
}
}
if port == 0 {
return nil, nil, errors.New("cannot define a port")
}
portStr = strconv.FormatInt(int64(port), 10)
for _, addr := range subset.Addresses {
svc.LoadBalancer.Servers = append(svc.LoadBalancer.Servers, dynamic.TCPServer{ svc.LoadBalancer.Servers = append(svc.LoadBalancer.Servers, dynamic.TCPServer{
Address: net.JoinHostPort(addr.IP, portStr), Address: net.JoinHostPort(address, strconv.Itoa(int(port))),
}) })
} }
} }
}
serviceName := provider.Normalize(service.Namespace + "-" + service.Name + "-" + portStr) serviceName := provider.Normalize(service.Namespace + "-" + service.Name + "-" + strconv.Itoa(int(svcPort.Port)))
services[serviceName] = &svc services[serviceName] = &svc
wrrSvc.Weighted.Services = append(wrrSvc.Weighted.Services, dynamic.TCPWRRService{Name: serviceName, Weight: &weight}) wrrSvc.Weighted.Services = append(wrrSvc.Weighted.Services, dynamic.TCPWRRService{Name: serviceName, Weight: &weight})

View file

@ -16,10 +16,12 @@ import (
"github.com/traefik/traefik/v3/pkg/types" "github.com/traefik/traefik/v3/pkg/types"
traefikversion "github.com/traefik/traefik/v3/pkg/version" traefikversion "github.com/traefik/traefik/v3/pkg/version"
corev1 "k8s.io/api/core/v1" corev1 "k8s.io/api/core/v1"
discoveryv1 "k8s.io/api/discovery/v1"
netv1 "k8s.io/api/networking/v1" netv1 "k8s.io/api/networking/v1"
kerror "k8s.io/apimachinery/pkg/api/errors" kerror "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/labels" "k8s.io/apimachinery/pkg/labels"
"k8s.io/apimachinery/pkg/selection"
kinformers "k8s.io/client-go/informers" kinformers "k8s.io/client-go/informers"
kclientset "k8s.io/client-go/kubernetes" kclientset "k8s.io/client-go/kubernetes"
"k8s.io/client-go/rest" "k8s.io/client-go/rest"
@ -41,7 +43,7 @@ type Client interface {
GetService(namespace, name string) (*corev1.Service, bool, error) GetService(namespace, name string) (*corev1.Service, bool, error)
GetSecret(namespace, name string) (*corev1.Secret, bool, error) GetSecret(namespace, name string) (*corev1.Secret, bool, error)
GetNodes() ([]*corev1.Node, bool, error) GetNodes() ([]*corev1.Node, bool, error)
GetEndpoints(namespace, name string) (*corev1.Endpoints, bool, error) GetEndpointSlicesForService(namespace, serviceName string) ([]*discoveryv1.EndpointSlice, error)
UpdateIngressStatus(ing *netv1.Ingress, ingStatus []netv1.IngressLoadBalancerIngress) error UpdateIngressStatus(ing *netv1.Ingress, ingStatus []netv1.IngressLoadBalancerIngress) error
} }
@ -185,7 +187,7 @@ func (c *clientWrapper) WatchAll(namespaces []string, stopCh <-chan struct{}) (<
if err != nil { if err != nil {
return nil, err return nil, err
} }
_, err = factoryKube.Core().V1().Endpoints().Informer().AddEventHandler(eventHandler) _, err = factoryKube.Discovery().V1().EndpointSlices().Informer().AddEventHandler(eventHandler)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -340,15 +342,20 @@ func (c *clientWrapper) GetService(namespace, name string) (*corev1.Service, boo
return service, exist, err return service, exist, err
} }
// GetEndpoints returns the named endpoints from the given namespace. // GetEndpointSlicesForService returns the EndpointSlices for the given service name in the given namespace.
func (c *clientWrapper) GetEndpoints(namespace, name string) (*corev1.Endpoints, bool, error) { func (c *clientWrapper) GetEndpointSlicesForService(namespace, serviceName string) ([]*discoveryv1.EndpointSlice, error) {
if !c.isWatchedNamespace(namespace) { if !c.isWatchedNamespace(namespace) {
return nil, false, fmt.Errorf("failed to get endpoints %s/%s: namespace is not within watched namespaces", namespace, name) return nil, fmt.Errorf("failed to get endpointslices for service %s/%s: namespace is not within watched namespaces", namespace, serviceName)
} }
endpoint, err := c.factoriesKube[c.lookupNamespace(namespace)].Core().V1().Endpoints().Lister().Endpoints(namespace).Get(name) serviceLabelRequirement, err := labels.NewRequirement(discoveryv1.LabelServiceName, selection.Equals, []string{serviceName})
exist, err := translateNotFoundError(err) if err != nil {
return endpoint, exist, err return nil, fmt.Errorf("failed to create service label selector requirement: %w", err)
}
serviceSelector := labels.NewSelector()
serviceSelector = serviceSelector.Add(*serviceLabelRequirement)
return c.factoriesKube[c.lookupNamespace(namespace)].Discovery().V1().EndpointSlices().Lister().EndpointSlices(namespace).List(serviceSelector)
} }
// GetSecret returns the named secret from the given namespace. // GetSecret returns the named secret from the given namespace.

View file

@ -6,6 +6,7 @@ import (
"github.com/traefik/traefik/v3/pkg/provider/kubernetes/k8s" "github.com/traefik/traefik/v3/pkg/provider/kubernetes/k8s"
corev1 "k8s.io/api/core/v1" corev1 "k8s.io/api/core/v1"
discoveryv1 "k8s.io/api/discovery/v1"
netv1 "k8s.io/api/networking/v1" netv1 "k8s.io/api/networking/v1"
) )
@ -15,13 +16,13 @@ type clientMock struct {
ingresses []*netv1.Ingress ingresses []*netv1.Ingress
services []*corev1.Service services []*corev1.Service
secrets []*corev1.Secret secrets []*corev1.Secret
endpoints []*corev1.Endpoints endpointSlices []*discoveryv1.EndpointSlice
nodes []*corev1.Node nodes []*corev1.Node
ingressClasses []*netv1.IngressClass ingressClasses []*netv1.IngressClass
apiServiceError error apiServiceError error
apiSecretError error apiSecretError error
apiEndpointsError error apiEndpointSlicesError error
apiNodesError error apiNodesError error
apiIngressStatusError error apiIngressStatusError error
@ -43,8 +44,8 @@ func newClientMock(path string) clientMock {
c.services = append(c.services, o) c.services = append(c.services, o)
case *corev1.Secret: case *corev1.Secret:
c.secrets = append(c.secrets, o) c.secrets = append(c.secrets, o)
case *corev1.Endpoints: case *discoveryv1.EndpointSlice:
c.endpoints = append(c.endpoints, o) c.endpointSlices = append(c.endpointSlices, o)
case *corev1.Node: case *corev1.Node:
c.nodes = append(c.nodes, o) c.nodes = append(c.nodes, o)
case *netv1.Ingress: case *netv1.Ingress:
@ -76,18 +77,19 @@ func (c clientMock) GetService(namespace, name string) (*corev1.Service, bool, e
return nil, false, c.apiServiceError return nil, false, c.apiServiceError
} }
func (c clientMock) GetEndpoints(namespace, name string) (*corev1.Endpoints, bool, error) { func (c clientMock) GetEndpointSlicesForService(namespace, serviceName string) ([]*discoveryv1.EndpointSlice, error) {
if c.apiEndpointsError != nil { if c.apiEndpointSlicesError != nil {
return nil, false, c.apiEndpointsError return nil, c.apiEndpointSlicesError
} }
for _, endpoints := range c.endpoints { var result []*discoveryv1.EndpointSlice
if endpoints.Namespace == namespace && endpoints.Name == name { for _, endpointSlice := range c.endpointSlices {
return endpoints, true, nil if endpointSlice.Namespace == namespace && endpointSlice.Labels[discoveryv1.LabelServiceName] == serviceName {
result = append(result, endpointSlice)
} }
} }
return &corev1.Endpoints{}, false, nil return result, nil
} }
func (c clientMock) GetNodes() ([]*corev1.Node, bool, error) { func (c clientMock) GetNodes() ([]*corev1.Node, bool, error) {

View file

@ -9,6 +9,7 @@ import (
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
corev1 "k8s.io/api/core/v1" corev1 "k8s.io/api/core/v1"
discoveryv1 "k8s.io/api/discovery/v1"
netv1 "k8s.io/api/networking/v1" netv1 "k8s.io/api/networking/v1"
kerror "k8s.io/apimachinery/pkg/api/errors" kerror "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
@ -188,10 +189,10 @@ func TestClientIgnoresHelmOwnedSecrets(t *testing.T) {
assert.False(t, found) assert.False(t, found)
} }
func TestClientIgnoresEmptyEndpointUpdates(t *testing.T) { func TestClientIgnoresEmptyEndpointSliceUpdates(t *testing.T) {
emptyEndpoint := &corev1.Endpoints{ emptyEndpointSlice := &discoveryv1.EndpointSlice{
ObjectMeta: metav1.ObjectMeta{ ObjectMeta: metav1.ObjectMeta{
Name: "empty-endpoint", Name: "empty-endpointslice",
Namespace: "test", Namespace: "test",
ResourceVersion: "1244", ResourceVersion: "1244",
Annotations: map[string]string{ Annotations: map[string]string{
@ -200,25 +201,31 @@ func TestClientIgnoresEmptyEndpointUpdates(t *testing.T) {
}, },
} }
filledEndpoint := &corev1.Endpoints{ samplePortName := "testing"
samplePortNumber := int32(1337)
samplePortProtocol := corev1.ProtocolTCP
sampleAddressReady := true
filledEndpointSlice := &discoveryv1.EndpointSlice{
ObjectMeta: metav1.ObjectMeta{ ObjectMeta: metav1.ObjectMeta{
Name: "filled-endpoint", Name: "filled-endpointslice",
Namespace: "test", Namespace: "test",
ResourceVersion: "1234", ResourceVersion: "1234",
}, },
Subsets: []corev1.EndpointSubset{{ AddressType: discoveryv1.AddressTypeIPv4,
Addresses: []corev1.EndpointAddress{{ Endpoints: []discoveryv1.Endpoint{{
IP: "10.13.37.1", Addresses: []string{"10.13.37.1"},
}}, Conditions: discoveryv1.EndpointConditions{
Ports: []corev1.EndpointPort{{ Ready: &sampleAddressReady,
Name: "testing", },
Port: 1337,
Protocol: "tcp",
}}, }},
Ports: []discoveryv1.EndpointPort{{
Name: &samplePortName,
Port: &samplePortNumber,
Protocol: &samplePortProtocol,
}}, }},
} }
kubeClient := kubefake.NewSimpleClientset(emptyEndpoint, filledEndpoint) kubeClient := kubefake.NewSimpleClientset(emptyEndpointSlice, filledEndpointSlice)
discovery, _ := kubeClient.Discovery().(*discoveryfake.FakeDiscovery) discovery, _ := kubeClient.Discovery().(*discoveryfake.FakeDiscovery)
discovery.FakedServerVersion = &kversion.Info{ discovery.FakedServerVersion = &kversion.Info{
@ -234,50 +241,72 @@ func TestClientIgnoresEmptyEndpointUpdates(t *testing.T) {
select { select {
case event := <-eventCh: case event := <-eventCh:
ep, ok := event.(*corev1.Endpoints) ep, ok := event.(*discoveryv1.EndpointSlice)
require.True(t, ok) require.True(t, ok)
assert.True(t, ep.Name == "empty-endpoint" || ep.Name == "filled-endpoint") assert.True(t, ep.Name == "empty-endpointslice" || ep.Name == "filled-endpointslice")
case <-time.After(50 * time.Millisecond): case <-time.After(50 * time.Millisecond):
assert.Fail(t, "expected to receive event for endpoints") assert.Fail(t, "expected to receive event for endpointslices")
} }
emptyEndpoint, err = kubeClient.CoreV1().Endpoints("test").Get(context.TODO(), "empty-endpoint", metav1.GetOptions{}) emptyEndpointSlice, err = kubeClient.DiscoveryV1().EndpointSlices("test").Get(context.TODO(), "empty-endpointslice", metav1.GetOptions{})
assert.NoError(t, err) assert.NoError(t, err)
// Update endpoint annotation and resource version (apparently not done by fake client itself) // Update endpoint annotation and resource version (apparently not done by fake client itself)
// to show an update that should not trigger an update event on our eventCh. // to show an update that should not trigger an update event on our eventCh.
// This reflects the behavior of kubernetes controllers which use endpoint annotations for leader election. // This reflects the behavior of kubernetes controllers which use endpoint annotations for leader election.
emptyEndpoint.Annotations["test-annotation"] = "___" emptyEndpointSlice.Annotations["test-annotation"] = "___"
emptyEndpoint.ResourceVersion = "1245" emptyEndpointSlice.ResourceVersion = "1245"
_, err = kubeClient.CoreV1().Endpoints("test").Update(context.TODO(), emptyEndpoint, metav1.UpdateOptions{}) _, err = kubeClient.DiscoveryV1().EndpointSlices("test").Update(context.TODO(), emptyEndpointSlice, metav1.UpdateOptions{})
require.NoError(t, err) require.NoError(t, err)
select { select {
case event := <-eventCh: case event := <-eventCh:
ep, ok := event.(*corev1.Endpoints) ep, ok := event.(*discoveryv1.EndpointSlice)
require.True(t, ok) require.True(t, ok)
assert.Fail(t, "didn't expect to receive event for empty endpoint update", ep.Name) assert.Fail(t, "didn't expect to receive event for empty endpointslice update", ep.Name)
case <-time.After(50 * time.Millisecond): case <-time.After(50 * time.Millisecond):
} }
filledEndpoint, err = kubeClient.CoreV1().Endpoints("test").Get(context.TODO(), "filled-endpoint", metav1.GetOptions{}) filledEndpointSlice, err = kubeClient.DiscoveryV1().EndpointSlices("test").Get(context.TODO(), "filled-endpointslice", metav1.GetOptions{})
assert.NoError(t, err) assert.NoError(t, err)
filledEndpoint.Subsets[0].Addresses[0].IP = "10.13.37.2" filledEndpointSlice.Endpoints[0].Addresses[0] = "10.13.37.2"
filledEndpoint.ResourceVersion = "1235" filledEndpointSlice.ResourceVersion = "1235"
_, err = kubeClient.CoreV1().Endpoints("test").Update(context.TODO(), filledEndpoint, metav1.UpdateOptions{}) _, err = kubeClient.DiscoveryV1().EndpointSlices("test").Update(context.TODO(), filledEndpointSlice, metav1.UpdateOptions{})
require.NoError(t, err) require.NoError(t, err)
select { select {
case event := <-eventCh: case event := <-eventCh:
ep, ok := event.(*corev1.Endpoints) ep, ok := event.(*discoveryv1.EndpointSlice)
require.True(t, ok) require.True(t, ok)
assert.Equal(t, "filled-endpoint", ep.Name) assert.Equal(t, "filled-endpointslice", ep.Name)
case <-time.After(50 * time.Millisecond): case <-time.After(50 * time.Millisecond):
assert.Fail(t, "expected to receive event for filled endpoint") assert.Fail(t, "expected to receive event for filled endpointslice")
}
select {
case <-eventCh:
assert.Fail(t, "received more than one event")
case <-time.After(50 * time.Millisecond):
}
newPortNumber := int32(42)
filledEndpointSlice.Ports[0].Port = &newPortNumber
filledEndpointSlice.ResourceVersion = "1236"
_, err = kubeClient.DiscoveryV1().EndpointSlices("test").Update(context.TODO(), filledEndpointSlice, metav1.UpdateOptions{})
require.NoError(t, err)
select {
case event := <-eventCh:
ep, ok := event.(*discoveryv1.EndpointSlice)
require.True(t, ok)
assert.Equal(t, "filled-endpointslice", ep.Name)
case <-time.After(50 * time.Millisecond):
assert.Fail(t, "expected to receive event for filled endpointslice")
} }
select { select {

View file

@ -1,32 +1,42 @@
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: tchouk - name: tchouk
port: 8089 port: 8089
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: toto namespace: toto
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.11.0.1
- ip: 10.11.0.2
ports:
- name: tchouk - name: tchouk
port: 8089 port: 8089
endpoints:
- addresses:
- 10.11.0.1
- 10.11.0.2
conditions:
ready: true
--- ---
kind: Ingress kind: Ingress

View file

@ -50,35 +50,41 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiversion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.30.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.41.0.1 - 10.30.0.1
ports: - 10.41.0.1
- port: 8080 conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiversion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service2 name: service2-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service2
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8080 conditions:
ready: true

View file

@ -40,18 +40,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8080 conditions:
ready: true

View file

@ -37,18 +37,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8080 conditions:
ready: true

View file

@ -30,18 +30,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8080 conditions:
ready: true

View file

@ -36,18 +36,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8080 conditions:
ready: true

View file

@ -52,15 +52,20 @@ spec:
externalName: "2001:0db8:3c4d:0015:0000:0000:1a2f:2a3b" externalName: "2001:0db8:3c4d:0015:0000:0000:1a2f:2a3b"
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service-bar name: service-bar-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service-bar
subsets: addressType: IPv6
- addresses: ports:
- ip: "2001:0db8:3c4d:0015:0000:0000:1a2f:1a2b"
ports:
- name: http - name: http
port: 8080 port: 8080
endpoints:
- addresses:
- "2001:0db8:3c4d:0015:0000:0000:1a2f:1a2b"
conditions:
ready: true

View file

@ -30,18 +30,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8443 - port: 8443
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8443 conditions:
ready: true

View file

@ -31,20 +31,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- name: https - name: https
port: 8443 port: 8443
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- name: https conditions:
port: 8443 ready: true

View file

@ -31,20 +31,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- name: https-foo - name: https-foo
port: 8443 port: 8443
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- name: https-foo conditions:
port: 8443 ready: true

View file

@ -29,18 +29,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8080 conditions:
ready: true

View file

@ -33,23 +33,42 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: tchouk - name: tchouk
port: 8089 port: 8089
endpoints:
- addresses: - addresses:
- ip: 10.10.0.1 - 10.10.0.1
- ip: 10.10.0.2 - 10.10.0.2
- ip: 10.10.0.3 conditions:
ports: ready: true
---
kind: EndpointSlice
apiVersion: discovery.k8s.io/v1
metadata:
name: service1-def
namespace: testing
labels:
kubernetes.io/service-name: service1
addressType: IPv4
ports:
- name: carotte - name: carotte
port: 8090 port: 8090
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
- 10.10.0.3
conditions:
ready: true

View file

@ -53,18 +53,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8080 conditions:
ready: true

View file

@ -41,18 +41,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8080 conditions:
ready: true

View file

@ -37,18 +37,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8080 conditions:
ready: true

View file

@ -31,14 +31,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -36,27 +36,38 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 80 - port: 80
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: defaultservice name: defaultservice-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: defaultservice
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -30,14 +30,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -28,14 +28,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -30,14 +30,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -30,14 +30,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -31,14 +31,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -37,15 +37,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -64,14 +64,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -29,14 +29,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -64,14 +64,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -29,15 +29,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- name: foobar - name: foobar
port: 4711 port: 4711
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -31,14 +31,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -31,15 +31,20 @@ spec:
type: ClusterIP type: ClusterIP
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: example-com name: example-com-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: example-com
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.11.0.1
ports:
- name: http - name: http
port: 80 port: 80
endpoints:
- addresses:
- 10.11.0.1
conditions:
ready: true

View file

@ -30,8 +30,14 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
addressType: IPv4
ports: null
endpoints: []

View file

@ -40,24 +40,43 @@ spec:
type: ClusterIP type: ClusterIP
--- ---
apiVersion: v1 kind: EndpointSlice
kind: Endpoints apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
subsets: labels:
- addresses: kubernetes.io/service-name: service1
- ip: 10.0.0.1
nodeName: admin.whoami.service1 addressType: IPv4
ports: ports:
- name: http-admin - name: http-admin
port: 8079 port: 8079
protocol: TCP protocol: TCP
endpoints:
- addresses: - addresses:
- ip: 10.0.0.1 - 10.0.0.1
nodeName: whoami.service1 conditions:
# targetRef: ready: true
ports: nodeName: admin.whoami.service1
---
kind: EndpointSlice
apiVersion: discovery.k8s.io/v1
metadata:
name: service1-def
namespace: testing
labels:
kubernetes.io/service-name: service1
addressType: IPv4
ports:
- name: http - name: http
port: 8080 port: 8080
protocol: TCP protocol: TCP
endpoints:
- addresses:
- 10.0.0.1
conditions:
ready: true
nodeName: whoami.service1

View file

@ -33,18 +33,23 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: carotte - name: carotte
port: 8090 port: 8090
- name: tchouk - name: tchouk
port: 8089 port: 8089
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true

View file

@ -33,24 +33,23 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- name: carotte - name: carotte
port: 8090 port: 8090
- name: tchouk - name: tchouk
port: 8089 port: 8089
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- name: carotte conditions:
port: 8090 ready: true
- name: tchouk
port: 8089

View file

@ -33,24 +33,23 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- name: carotte - name: carotte
port: 8090 port: 8090
- name: tchouk - name: tchouk
port: 8089 port: 8089
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- name: carotte conditions:
port: 8090 ready: true
- name: tchouk
port: 8089

View file

@ -28,14 +28,20 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses:
- 10.10.0.1
conditions:
ready: true

View file

@ -38,18 +38,21 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8080 conditions:
ready: true

View file

@ -40,18 +40,23 @@ spec:
clusterIP: 10.0.0.1 clusterIP: 10.0.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
- ip: 10.10.0.2
ports:
- name: carotte - name: carotte
port: 8090 port: 8090
- name: tchouk - name: tchouk
port: 8089 port: 8089
endpoints:
- addresses:
- 10.10.0.1
- 10.10.0.2
conditions:
ready: true

View file

@ -52,35 +52,41 @@ spec:
clusterIP: 10.1.0.1 clusterIP: 10.1.0.1
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service1 name: service1-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service1
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.1
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.1 - 10.10.0.1
ports: - 10.21.0.1
- port: 8080 conditions:
ready: true
--- ---
kind: Endpoints kind: EndpointSlice
apiVersion: v1 apiVersion: discovery.k8s.io/v1
metadata: metadata:
name: service2 name: service2-abc
namespace: testing namespace: testing
labels:
kubernetes.io/service-name: service2
subsets: addressType: IPv4
- addresses: ports:
- ip: 10.10.0.2
ports:
- port: 8080 - port: 8080
name: ""
endpoints:
- addresses: - addresses:
- ip: 10.21.0.2 - 10.10.0.2
ports: - 10.21.0.2
- port: 8080 conditions:
ready: true

Some files were not shown because too many files have changed in this diff Show more