mirror of
https://github.com/kubernetes/client-go.git
synced 2025-06-24 14:12:18 +00:00
client-go/util/watchlist: intro CanUseWatchListForListRequest
Kubernetes-commit: 38fae9b799393f6fe17d07fb8148f05b1110859b
This commit is contained in:
parent
03443e7ede
commit
9a760efea1
82
util/watchlist/watch_list.go
Normal file
82
util/watchlist/watch_list.go
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2024 The Kubernetes Authors.
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package watchlist
|
||||||
|
|
||||||
|
import (
|
||||||
|
metainternalversion "k8s.io/apimachinery/pkg/apis/meta/internalversion"
|
||||||
|
metainternalversionvalidation "k8s.io/apimachinery/pkg/apis/meta/internalversion/validation"
|
||||||
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
|
"k8s.io/apimachinery/pkg/runtime"
|
||||||
|
utilruntime "k8s.io/apimachinery/pkg/util/runtime"
|
||||||
|
clientfeatures "k8s.io/client-go/features"
|
||||||
|
"k8s.io/utils/ptr"
|
||||||
|
)
|
||||||
|
|
||||||
|
var scheme = runtime.NewScheme()
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
utilruntime.Must(metainternalversion.AddToScheme(scheme))
|
||||||
|
}
|
||||||
|
|
||||||
|
// PrepareWatchListOptionsFromListOptions creates a new ListOptions
|
||||||
|
// that can be used for a watch-list request from the given listOptions.
|
||||||
|
//
|
||||||
|
// This function also determines if the given listOptions can be used to form a watch-list request,
|
||||||
|
// which would result in streaming semantically equivalent data from the server.
|
||||||
|
func PrepareWatchListOptionsFromListOptions(listOptions metav1.ListOptions) (metav1.ListOptions, bool, error) {
|
||||||
|
if !clientfeatures.FeatureGates().Enabled(clientfeatures.WatchListClient) {
|
||||||
|
return metav1.ListOptions{}, false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
internalListOptions := &metainternalversion.ListOptions{}
|
||||||
|
if err := scheme.Convert(&listOptions, internalListOptions, nil); err != nil {
|
||||||
|
return metav1.ListOptions{}, false, err
|
||||||
|
}
|
||||||
|
if errs := metainternalversionvalidation.ValidateListOptions(internalListOptions, true); len(errs) > 0 {
|
||||||
|
return metav1.ListOptions{}, false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
watchListOptions := listOptions
|
||||||
|
// this is our legacy case, the cache ignores LIMIT for
|
||||||
|
// ResourceVersion == 0 and RVM=unset|NotOlderThan
|
||||||
|
if listOptions.Limit > 0 && listOptions.ResourceVersion != "0" {
|
||||||
|
return metav1.ListOptions{}, false, nil
|
||||||
|
}
|
||||||
|
watchListOptions.Limit = 0
|
||||||
|
|
||||||
|
// to ensure that we can create a watch-list request that returns
|
||||||
|
// semantically equivalent data for the given listOptions,
|
||||||
|
// we need to validate that the RVM for the list is supported by watch-list requests.
|
||||||
|
if listOptions.ResourceVersionMatch == metav1.ResourceVersionMatchExact {
|
||||||
|
return metav1.ListOptions{}, false, nil
|
||||||
|
}
|
||||||
|
watchListOptions.ResourceVersionMatch = metav1.ResourceVersionMatchNotOlderThan
|
||||||
|
|
||||||
|
watchListOptions.Watch = true
|
||||||
|
watchListOptions.AllowWatchBookmarks = true
|
||||||
|
watchListOptions.SendInitialEvents = ptr.To(true)
|
||||||
|
|
||||||
|
internalWatchListOptions := &metainternalversion.ListOptions{}
|
||||||
|
if err := scheme.Convert(&watchListOptions, internalWatchListOptions, nil); err != nil {
|
||||||
|
return metav1.ListOptions{}, false, err
|
||||||
|
}
|
||||||
|
if errs := metainternalversionvalidation.ValidateListOptions(internalWatchListOptions, true); len(errs) > 0 {
|
||||||
|
return metav1.ListOptions{}, false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return watchListOptions, true, nil
|
||||||
|
}
|
200
util/watchlist/watch_list_test.go
Normal file
200
util/watchlist/watch_list_test.go
Normal file
@ -0,0 +1,200 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2024 The Kubernetes Authors.
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package watchlist
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
|
||||||
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
|
clientfeatures "k8s.io/client-go/features"
|
||||||
|
clientfeaturestesting "k8s.io/client-go/features/testing"
|
||||||
|
"k8s.io/utils/ptr"
|
||||||
|
)
|
||||||
|
|
||||||
|
// TestPrepareWatchListOptionsFromListOptions test the following cases:
|
||||||
|
//
|
||||||
|
// +--------------------------+-----------------+---------+-----------------+
|
||||||
|
// | ResourceVersionMatch | ResourceVersion | Limit | Continuation |
|
||||||
|
// +--------------------------+-----------------+---------+-----------------+
|
||||||
|
// | unset/NotOlderThan/Exact | unset/0/100 | unset/4 | unset/FakeToken |
|
||||||
|
// +--------------------------+-----------------+---------+-----------------+
|
||||||
|
func TestPrepareWatchListOptionsFromListOptions(t *testing.T) {
|
||||||
|
scenarios := []struct {
|
||||||
|
name string
|
||||||
|
listOptions metav1.ListOptions
|
||||||
|
enableWatchListFG bool
|
||||||
|
|
||||||
|
expectToPrepareWatchListOptions bool
|
||||||
|
expectedWatchListOptions metav1.ListOptions
|
||||||
|
}{
|
||||||
|
|
||||||
|
{
|
||||||
|
name: "can't enable watch list for: WatchListClient=off, RVM=unset, RV=unset, Limit=unset, Continuation=unset",
|
||||||
|
enableWatchListFG: false,
|
||||||
|
expectToPrepareWatchListOptions: false,
|
||||||
|
},
|
||||||
|
// +----------------------+-----------------+-------+--------------+
|
||||||
|
// | ResourceVersionMatch | ResourceVersion | Limit | Continuation |
|
||||||
|
// +----------------------+-----------------+-------+--------------+
|
||||||
|
// | unset | unset | unset | unset |
|
||||||
|
// | unset | 0 | unset | unset |
|
||||||
|
// | unset | 100 | unset | unset |
|
||||||
|
// | unset | 0 | 4 | unset |
|
||||||
|
// | unset | 0 | unset | FakeToken |
|
||||||
|
// +----------------------+-----------------+-------+--------------+
|
||||||
|
{
|
||||||
|
name: "can enable watch list for: RVM=unset, RV=unset, Limit=unset, Continuation=unset",
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: true,
|
||||||
|
expectedWatchListOptions: expectedWatchListOptionsFor(""),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "can enable watch list for: RVM=unset, RV=0, Limit=unset, Continuation=unset",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersion: "0"},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: true,
|
||||||
|
expectedWatchListOptions: expectedWatchListOptionsFor("0"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "can enable watch list for: RVM=unset, RV=100, Limit=unset, Continuation=unset",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersion: "100"},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: true,
|
||||||
|
expectedWatchListOptions: expectedWatchListOptionsFor("100"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "legacy: can enable watch list for: RVM=unset, RV=0, Limit=4, Continuation=unset",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersion: "0", Limit: 4},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: true,
|
||||||
|
expectedWatchListOptions: expectedWatchListOptionsFor("0"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "can't enable watch list for: RVM=unset, RV=0, Limit=unset, Continuation=FakeToken",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersion: "0", Continue: "FakeToken"},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: false,
|
||||||
|
},
|
||||||
|
// +----------------------+-----------------+-------+--------------+
|
||||||
|
// | ResourceVersionMatch | ResourceVersion | Limit | Continuation |
|
||||||
|
// +----------------------+-----------------+-------+--------------+
|
||||||
|
// | NotOlderThan | unset | unset | unset |
|
||||||
|
// | NotOlderThan | 0 | unset | unset |
|
||||||
|
// | NotOlderThan | 100 | unset | unset |
|
||||||
|
// | NotOlderThan | 0 | 4 | unset |
|
||||||
|
// | NotOlderThan | 0 | unset | FakeToken |
|
||||||
|
// +----------------------+-----------------+-------+--------------+
|
||||||
|
{
|
||||||
|
name: "can't enable watch list for: RVM=NotOlderThan, RV=unset, Limit=unset, Continuation=unset",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersionMatch: metav1.ResourceVersionMatchNotOlderThan},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "can enable watch list for: RVM=NotOlderThan, RV=0, Limit=unset, Continuation=unset",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersionMatch: metav1.ResourceVersionMatchNotOlderThan, ResourceVersion: "0"},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: true,
|
||||||
|
expectedWatchListOptions: expectedWatchListOptionsFor("0"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "can enable watch list for: RVM=NotOlderThan, RV=100, Limit=unset, Continuation=unset",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersionMatch: metav1.ResourceVersionMatchNotOlderThan, ResourceVersion: "100"},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: true,
|
||||||
|
expectedWatchListOptions: expectedWatchListOptionsFor("100"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "legacy: can enable watch list for: RVM=NotOlderThan, RV=0, Limit=4, Continuation=unset",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersionMatch: metav1.ResourceVersionMatchNotOlderThan, ResourceVersion: "0", Limit: 4},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: true,
|
||||||
|
expectedWatchListOptions: expectedWatchListOptionsFor("0"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "can't enable watch list for: RVM=NotOlderThan, RV=0, Limit=unset, Continuation=FakeToken",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersionMatch: metav1.ResourceVersionMatchNotOlderThan, ResourceVersion: "0", Continue: "FakeToken"},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: false,
|
||||||
|
},
|
||||||
|
|
||||||
|
// +----------------------+-----------------+-------+--------------+
|
||||||
|
// | ResourceVersionMatch | ResourceVersion | Limit | Continuation |
|
||||||
|
// +----------------------+-----------------+-------+--------------+
|
||||||
|
// | Exact | unset | unset | unset |
|
||||||
|
// | Exact | 0 | unset | unset |
|
||||||
|
// | Exact | 100 | unset | unset |
|
||||||
|
// | Exact | 0 | 4 | unset |
|
||||||
|
// | Exact | 0 | unset | FakeToken |
|
||||||
|
// +----------------------+-----------------+-------+--------------+
|
||||||
|
{
|
||||||
|
name: "can't enable watch list for: RVM=Exact, RV=unset, Limit=unset, Continuation=unset",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersionMatch: metav1.ResourceVersionMatchExact},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "can enable watch list for: RVM=Exact, RV=0, Limit=unset, Continuation=unset",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersionMatch: metav1.ResourceVersionMatchExact, ResourceVersion: "0"},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "can enable watch list for: RVM=Exact, RV=100, Limit=unset, Continuation=unset",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersionMatch: metav1.ResourceVersionMatchExact, ResourceVersion: "100"},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "can't enable watch list for: RVM=Exact, RV=0, Limit=4, Continuation=unset",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersionMatch: metav1.ResourceVersionMatchExact, ResourceVersion: "0", Limit: 4},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "can't enable watch list for: RVM=Exact, RV=0, Limit=unset, Continuation=FakeToken",
|
||||||
|
listOptions: metav1.ListOptions{ResourceVersionMatch: metav1.ResourceVersionMatchExact, ResourceVersion: "0", Continue: "FakeToken"},
|
||||||
|
enableWatchListFG: true,
|
||||||
|
expectToPrepareWatchListOptions: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, scenario := range scenarios {
|
||||||
|
t.Run(scenario.name, func(t *testing.T) {
|
||||||
|
clientfeaturestesting.SetFeatureDuringTest(t, clientfeatures.WatchListClient, scenario.enableWatchListFG)
|
||||||
|
|
||||||
|
watchListOptions, hasWatchListOptionsPrepared, err := PrepareWatchListOptionsFromListOptions(scenario.listOptions)
|
||||||
|
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, scenario.expectToPrepareWatchListOptions, hasWatchListOptionsPrepared)
|
||||||
|
require.Equal(t, scenario.expectedWatchListOptions, watchListOptions)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func expectedWatchListOptionsFor(rv string) metav1.ListOptions {
|
||||||
|
var watchListOptions metav1.ListOptions
|
||||||
|
|
||||||
|
watchListOptions.ResourceVersion = rv
|
||||||
|
watchListOptions.ResourceVersionMatch = metav1.ResourceVersionMatchNotOlderThan
|
||||||
|
watchListOptions.Watch = true
|
||||||
|
watchListOptions.AllowWatchBookmarks = true
|
||||||
|
watchListOptions.SendInitialEvents = ptr.To(true)
|
||||||
|
|
||||||
|
return watchListOptions
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user