Skip to content

Commit b40de38

Browse files
committed
[Conformance]: Adds GatewayClass SupportedVersion Test
Adds a test to ensure implementations conform to the GatewayClass SupportedVersion status condition. Signed-off-by: Daneyon Hansen <[email protected]>
1 parent 9f7dd62 commit b40de38

File tree

4 files changed

+207
-0
lines changed

4 files changed

+207
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,103 @@
1+
/*
2+
Copyright 2024 The Kubernetes Authors.
3+
4+
Licensed under the Apache License, Version 2.0 (the "License");
5+
you may not use this file except in compliance with the License.
6+
You may obtain a copy of the License at
7+
8+
http://www.apache.org/licenses/LICENSE-2.0
9+
10+
Unless required by applicable law or agreed to in writing, software
11+
distributed under the License is distributed on an "AS IS" BASIS,
12+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
See the License for the specific language governing permissions and
14+
limitations under the License.
15+
*/
16+
17+
package tests
18+
19+
import (
20+
"context"
21+
"testing"
22+
23+
"github.com/stretchr/testify/require"
24+
apiextv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
25+
"k8s.io/apimachinery/pkg/types"
26+
27+
"sigs.k8s.io/gateway-api/conformance/utils/kubernetes"
28+
"sigs.k8s.io/gateway-api/conformance/utils/suite"
29+
"sigs.k8s.io/gateway-api/pkg/consts"
30+
"sigs.k8s.io/gateway-api/pkg/features"
31+
)
32+
33+
func init() {
34+
ConformanceTests = append(ConformanceTests, GatewayClassSupportedVersionCondition)
35+
}
36+
37+
var GatewayClassSupportedVersionCondition = suite.ConformanceTest{
38+
ShortName: "GatewayClassSupportedVersionCondition",
39+
Features: []features.FeatureName{
40+
features.SupportGateway,
41+
},
42+
Description: "A GatewayClass should set the SupportedVersion condition based on the presence and version of Gateway API CRDs in the cluster",
43+
Manifests: []string{"tests/gatewayclass-supported-version-condition.yaml"},
44+
Test: func(t *testing.T, s *suite.ConformanceTestSuite) {
45+
gwc := types.NamespacedName{Name: "gatewayclass-supported-version-condition"}
46+
47+
t.Run("SupportedVersion condition should be set correctly", func(t *testing.T) {
48+
ctx, cancel := context.WithTimeout(context.Background(), s.TimeoutConfig.DefaultTestTimeout)
49+
defer cancel()
50+
51+
// Ensure GatewayClass conditions are as expected before proceeding
52+
kubernetes.GWCMustHaveAcceptedConditionTrue(t, s.Client, s.TimeoutConfig, gwc.Name)
53+
kubernetes.GWCMustHaveSupportedVersionConditionAny(t, s.Client, s.TimeoutConfig, gwc.Name)
54+
55+
// Retrieve the GatewayClass CRD
56+
crd := &apiextv1.CustomResourceDefinition{}
57+
crdName := types.NamespacedName{Name: "gatewayclasses.gateway.networking.k8s.io"}
58+
err := s.Client.Get(ctx, crdName, crd)
59+
require.NoErrorf(t, err, "error getting GatewayClass CRD: %v", err)
60+
61+
if crd.Annotations != nil {
62+
// Remove the bundle version annotation if it exists
63+
if _, exists := crd.Annotations[consts.BundleVersionAnnotation]; !exists {
64+
t.Fatalf("Annotation %q does not exist on CRD %s", consts.BundleVersionAnnotation, crdName)
65+
}
66+
delete(crd.Annotations, consts.BundleVersionAnnotation)
67+
if err := s.Client.Update(ctx, crd); err != nil {
68+
t.Fatalf("Failed to update CRD %s: %v", crdName, err)
69+
}
70+
}
71+
72+
// Ensure the SupportedVersion status condition is false
73+
kubernetes.GWCMustHaveSupportedVersionConditionFalse(t, s.Client, s.TimeoutConfig, gwc.Name)
74+
75+
// Add the bundle version annotation
76+
crd.Annotations[consts.BundleVersionAnnotation] = consts.BundleVersion
77+
if err := s.Client.Update(ctx, crd); err != nil {
78+
t.Fatalf("Failed to update CRD %s: %v", crdName, err)
79+
}
80+
81+
// Ensure the SupportedVersion status condition is true
82+
kubernetes.GWCMustHaveSupportedVersionConditionTrue(t, s.Client, s.TimeoutConfig, gwc.Name)
83+
84+
// Set the bundle version annotation to an unsupported version
85+
crd.Annotations[consts.BundleVersionAnnotation] = "v0.0.0"
86+
if err := s.Client.Update(ctx, crd); err != nil {
87+
t.Fatalf("Failed to update CRD %s: %v", crdName, err)
88+
}
89+
90+
// Ensure the SupportedVersion is false
91+
kubernetes.GWCMustHaveSupportedVersionConditionFalse(t, s.Client, s.TimeoutConfig, gwc.Name)
92+
93+
// Add the bundle version annotation back
94+
crd.Annotations[consts.BundleVersionAnnotation] = consts.BundleVersion
95+
if err := s.Client.Update(ctx, crd); err != nil {
96+
t.Fatalf("Failed to update CRD %s: %v", crdName, err)
97+
}
98+
99+
// Ensure the SupportedVersion is true
100+
kubernetes.GWCMustHaveSupportedVersionConditionTrue(t, s.Client, s.TimeoutConfig, gwc.Name)
101+
})
102+
},
103+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
apiVersion: gateway.networking.k8s.io/v1
2+
kind: GatewayClass
3+
metadata:
4+
name: gatewayclass-supported-version-condition
5+
spec:
6+
controllerName: "{GATEWAY_CONTROLLER_NAME}"

conformance/utils/config/timeout.go

+8
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,10 @@ type TimeoutConfig struct {
5656
// Max value for conformant implementation: None
5757
GWCMustBeAccepted time.Duration
5858

59+
// GWCMustBeSupportedVersion represents the maximum time for a GatewayClass to have a SupportedVersion condition set to true.
60+
// Max value for conformant implementation: None
61+
GWCMustBeSupportedVersion time.Duration
62+
5963
// HTTPRouteMustNotHaveParents represents the maximum time for an HTTPRoute to have either no parents or a single parent that is not accepted.
6064
// Max value for conformant implementation: None
6165
HTTPRouteMustNotHaveParents time.Duration
@@ -115,6 +119,7 @@ func DefaultTimeoutConfig() TimeoutConfig {
115119
GatewayStatusMustHaveListeners: 60 * time.Second,
116120
GatewayListenersMustHaveConditions: 60 * time.Second,
117121
GWCMustBeAccepted: 180 * time.Second,
122+
GWCMustBeSupportedVersion: 180 * time.Second,
118123
HTTPRouteMustNotHaveParents: 60 * time.Second,
119124
HTTPRouteMustHaveCondition: 60 * time.Second,
120125
TLSRouteMustHaveCondition: 60 * time.Second,
@@ -155,6 +160,9 @@ func SetupTimeoutConfig(timeoutConfig *TimeoutConfig) {
155160
if timeoutConfig.GWCMustBeAccepted == 0 {
156161
timeoutConfig.GWCMustBeAccepted = defaultTimeoutConfig.GWCMustBeAccepted
157162
}
163+
if timeoutConfig.GWCMustBeSupportedVersion == 0 {
164+
timeoutConfig.GWCMustBeSupportedVersion = defaultTimeoutConfig.GWCMustBeSupportedVersion
165+
}
158166
if timeoutConfig.HTTPRouteMustNotHaveParents == 0 {
159167
timeoutConfig.HTTPRouteMustNotHaveParents = defaultTimeoutConfig.HTTPRouteMustNotHaveParents
160168
}

conformance/utils/kubernetes/helpers.go

+90
Original file line numberDiff line numberDiff line change
@@ -77,11 +77,26 @@ func GWCMustHaveAcceptedConditionTrue(t *testing.T, c client.Client, timeoutConf
7777
return gwcMustBeAccepted(t, c, timeoutConfig, gwcName, string(metav1.ConditionTrue))
7878
}
7979

80+
// GWCMustHaveSupportedVersionConditionTrue waits until the specified GatewayClass has a SupportedVersion condition set with a status value equal to True.
81+
func GWCMustHaveSupportedVersionConditionTrue(t *testing.T, c client.Client, timeoutConfig config.TimeoutConfig, gwcName string) string {
82+
return gwcMustBeSupportedVersion(t, c, timeoutConfig, gwcName, string(metav1.ConditionTrue))
83+
}
84+
85+
// GWCMustHaveSupportedVersionConditionFalse waits until the specified GatewayClass has a SupportedVersion condition set with a status value equal to False.
86+
func GWCMustHaveSupportedVersionConditionFalse(t *testing.T, c client.Client, timeoutConfig config.TimeoutConfig, gwcName string) string {
87+
return gwcMustBeSupportedVersion(t, c, timeoutConfig, gwcName, string(metav1.ConditionFalse))
88+
}
89+
8090
// GWCMustHaveAcceptedConditionAny waits until the specified GatewayClass has an Accepted condition set with a status set to any value.
8191
func GWCMustHaveAcceptedConditionAny(t *testing.T, c client.Client, timeoutConfig config.TimeoutConfig, gwcName string) string {
8292
return gwcMustBeAccepted(t, c, timeoutConfig, gwcName, "")
8393
}
8494

95+
// GWCMustHaveSupportedVersionConditionAny waits until the specified GatewayClass has a SupportedVersion condition set to any value.
96+
func GWCMustHaveSupportedVersionConditionAny(t *testing.T, c client.Client, timeoutConfig config.TimeoutConfig, gwcName string) string {
97+
return gwcMustBeSupportedVersion(t, c, timeoutConfig, gwcName, "")
98+
}
99+
85100
// gwcMustBeAccepted waits until the specified GatewayClass has an Accepted
86101
// condition set. Passing an empty status string means that any value
87102
// will be accepted. It also returns the ControllerName for the GatewayClass.
@@ -112,6 +127,35 @@ func gwcMustBeAccepted(t *testing.T, c client.Client, timeoutConfig config.Timeo
112127
return controllerName
113128
}
114129

130+
// gwcMustBeSupportedVersion waits until the specified GatewayClass has a SupportedVersion condition set.
131+
// Passing an empty status string means that any value will be accepted. It also returns the ControllerName
132+
// for the GatewayClass. This will cause the test to halt if the specified timeout is exceeded.
133+
func gwcMustBeSupportedVersion(t *testing.T, c client.Client, timeoutConfig config.TimeoutConfig, gwcName, expectedStatus string) string {
134+
t.Helper()
135+
136+
var controllerName string
137+
waitErr := wait.PollUntilContextTimeout(context.Background(), 1*time.Second, timeoutConfig.GWCMustBeSupportedVersion, true, func(ctx context.Context) (bool, error) {
138+
gwc := &gatewayv1.GatewayClass{}
139+
err := c.Get(ctx, types.NamespacedName{Name: gwcName}, gwc)
140+
if err != nil {
141+
return false, fmt.Errorf("error fetching GatewayClass: %w", err)
142+
}
143+
144+
controllerName = string(gwc.Spec.ControllerName)
145+
146+
if err := ConditionsHaveLatestObservedGeneration(gwc, gwc.Status.Conditions); err != nil {
147+
tlog.Log(t, "GatewayClass", err)
148+
return false, nil
149+
}
150+
151+
// Passing an empty string as the Reason means that any Reason will do.
152+
return findConditionInList(t, gwc.Status.Conditions, "SupportedVersion", expectedStatus, ""), nil
153+
})
154+
require.NoErrorf(t, waitErr, "error waiting for %s GatewayClass to have SupportedVersion condition to be set: %v", gwcName, waitErr)
155+
156+
return controllerName
157+
}
158+
115159
// GatewayMustHaveLatestConditions waits until the specified Gateway has
116160
// all conditions updated with the latest observed generation.
117161
func GatewayMustHaveLatestConditions(t *testing.T, c client.Client, timeoutConfig config.TimeoutConfig, gwNN types.NamespacedName) {
@@ -249,6 +293,52 @@ func NamespacesMustBeReady(t *testing.T, c client.Client, timeoutConfig config.T
249293
require.NoErrorf(t, waitErr, "error waiting for %s namespaces to be ready", strings.Join(namespaces, ", "))
250294
}
251295

296+
// GatewayClassMustHaveCondition checks that the supplied GatewayClass has the supplied Condition,
297+
// halting after the specified timeout is exceeded.
298+
func GatewayClassMustHaveCondition(
299+
t *testing.T,
300+
client client.Client,
301+
timeoutConfig config.TimeoutConfig,
302+
gcName string,
303+
expectedCondition metav1.Condition,
304+
) {
305+
t.Helper()
306+
307+
waitErr := wait.PollUntilContextTimeout(
308+
context.Background(),
309+
1*time.Second,
310+
timeoutConfig.GWCMustBeSupportedVersion,
311+
true,
312+
func(ctx context.Context) (bool, error) {
313+
gc := &gatewayv1.GatewayClass{}
314+
gcNN := types.NamespacedName{
315+
Name: gcName,
316+
}
317+
err := client.Get(ctx, gcNN, gc)
318+
if err != nil {
319+
return false, fmt.Errorf("error fetching GatewayClass: %w", err)
320+
}
321+
322+
if err := ConditionsHaveLatestObservedGeneration(gc, gc.Status.Conditions); err != nil {
323+
return false, err
324+
}
325+
326+
if findConditionInList(t,
327+
gc.Status.Conditions,
328+
expectedCondition.Type,
329+
string(expectedCondition.Status),
330+
expectedCondition.Reason,
331+
) {
332+
return true, nil
333+
}
334+
335+
return false, nil
336+
},
337+
)
338+
339+
require.NoErrorf(t, waitErr, "error waiting for GatewayClass status to have a Condition matching expectations")
340+
}
341+
252342
// GatewayMustHaveCondition checks that the supplied Gateway has the supplied Condition,
253343
// halting after the specified timeout is exceeded.
254344
func GatewayMustHaveCondition(

0 commit comments

Comments
 (0)