mirror of https://github.com/k3s-io/k3s
231 lines
7.0 KiB
Go
231 lines
7.0 KiB
Go
/*
|
|
Copyright 2015 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 tests
|
|
|
|
import (
|
|
"net/http/httptest"
|
|
"net/url"
|
|
"testing"
|
|
"time"
|
|
|
|
"k8s.io/api/core/v1"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/fields"
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
"k8s.io/apimachinery/pkg/watch"
|
|
restclient "k8s.io/client-go/rest"
|
|
. "k8s.io/client-go/tools/cache"
|
|
watchtools "k8s.io/client-go/tools/watch"
|
|
utiltesting "k8s.io/client-go/util/testing"
|
|
"k8s.io/kubernetes/pkg/api/testapi"
|
|
clientset "k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset"
|
|
)
|
|
|
|
func parseSelectorOrDie(s string) fields.Selector {
|
|
selector, err := fields.ParseSelector(s)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return selector
|
|
}
|
|
|
|
// buildQueryValues is a convenience function for knowing if a namespace should be in a query param or not
|
|
func buildQueryValues(query url.Values) url.Values {
|
|
v := url.Values{}
|
|
if query != nil {
|
|
for key, values := range query {
|
|
for _, value := range values {
|
|
v.Add(key, value)
|
|
}
|
|
}
|
|
}
|
|
return v
|
|
}
|
|
|
|
func buildLocation(resourcePath string, query url.Values) string {
|
|
return resourcePath + "?" + query.Encode()
|
|
}
|
|
|
|
func TestListWatchesCanList(t *testing.T) {
|
|
fieldSelectorQueryParamName := metav1.FieldSelectorQueryParam("v1")
|
|
table := []struct {
|
|
location string
|
|
resource string
|
|
namespace string
|
|
fieldSelector fields.Selector
|
|
}{
|
|
// Node
|
|
{
|
|
location: testapi.Default.ResourcePath("nodes", metav1.NamespaceAll, ""),
|
|
resource: "nodes",
|
|
namespace: metav1.NamespaceAll,
|
|
fieldSelector: parseSelectorOrDie(""),
|
|
},
|
|
// pod with "assigned" field selector.
|
|
{
|
|
location: buildLocation(
|
|
testapi.Default.ResourcePath("pods", metav1.NamespaceAll, ""),
|
|
buildQueryValues(url.Values{fieldSelectorQueryParamName: []string{"spec.host="}})),
|
|
resource: "pods",
|
|
namespace: metav1.NamespaceAll,
|
|
fieldSelector: fields.Set{"spec.host": ""}.AsSelector(),
|
|
},
|
|
// pod in namespace "foo"
|
|
{
|
|
location: buildLocation(
|
|
testapi.Default.ResourcePath("pods", "foo", ""),
|
|
buildQueryValues(url.Values{fieldSelectorQueryParamName: []string{"spec.host="}})),
|
|
resource: "pods",
|
|
namespace: "foo",
|
|
fieldSelector: fields.Set{"spec.host": ""}.AsSelector(),
|
|
},
|
|
}
|
|
for _, item := range table {
|
|
handler := utiltesting.FakeHandler{
|
|
StatusCode: 500,
|
|
ResponseBody: "",
|
|
T: t,
|
|
}
|
|
server := httptest.NewServer(&handler)
|
|
defer server.Close()
|
|
client := clientset.NewForConfigOrDie(&restclient.Config{Host: server.URL, ContentConfig: restclient.ContentConfig{GroupVersion: &schema.GroupVersion{Group: "", Version: "v1"}}})
|
|
lw := NewListWatchFromClient(client.Core().RESTClient(), item.resource, item.namespace, item.fieldSelector)
|
|
lw.DisableChunking = true
|
|
// This test merely tests that the correct request is made.
|
|
lw.List(metav1.ListOptions{})
|
|
handler.ValidateRequest(t, item.location, "GET", nil)
|
|
}
|
|
}
|
|
|
|
func TestListWatchesCanWatch(t *testing.T) {
|
|
fieldSelectorQueryParamName := metav1.FieldSelectorQueryParam("v1")
|
|
table := []struct {
|
|
rv string
|
|
location string
|
|
resource string
|
|
namespace string
|
|
fieldSelector fields.Selector
|
|
}{
|
|
// Node
|
|
{
|
|
location: buildLocation(
|
|
testapi.Default.ResourcePath("nodes", metav1.NamespaceAll, ""),
|
|
buildQueryValues(url.Values{"watch": []string{"true"}})),
|
|
rv: "",
|
|
resource: "nodes",
|
|
namespace: metav1.NamespaceAll,
|
|
fieldSelector: parseSelectorOrDie(""),
|
|
},
|
|
{
|
|
location: buildLocation(
|
|
testapi.Default.ResourcePath("nodes", metav1.NamespaceAll, ""),
|
|
buildQueryValues(url.Values{"resourceVersion": []string{"42"}, "watch": []string{"true"}})),
|
|
rv: "42",
|
|
resource: "nodes",
|
|
namespace: metav1.NamespaceAll,
|
|
fieldSelector: parseSelectorOrDie(""),
|
|
},
|
|
// pod with "assigned" field selector.
|
|
{
|
|
location: buildLocation(
|
|
testapi.Default.ResourcePath("pods", metav1.NamespaceAll, ""),
|
|
buildQueryValues(url.Values{fieldSelectorQueryParamName: []string{"spec.host="}, "resourceVersion": []string{"0"}, "watch": []string{"true"}})),
|
|
rv: "0",
|
|
resource: "pods",
|
|
namespace: metav1.NamespaceAll,
|
|
fieldSelector: fields.Set{"spec.host": ""}.AsSelector(),
|
|
},
|
|
// pod with namespace foo and assigned field selector
|
|
{
|
|
location: buildLocation(
|
|
testapi.Default.ResourcePath("pods", "foo", ""),
|
|
buildQueryValues(url.Values{fieldSelectorQueryParamName: []string{"spec.host="}, "resourceVersion": []string{"0"}, "watch": []string{"true"}})),
|
|
rv: "0",
|
|
resource: "pods",
|
|
namespace: "foo",
|
|
fieldSelector: fields.Set{"spec.host": ""}.AsSelector(),
|
|
},
|
|
}
|
|
|
|
for _, item := range table {
|
|
handler := utiltesting.FakeHandler{
|
|
StatusCode: 500,
|
|
ResponseBody: "",
|
|
T: t,
|
|
}
|
|
server := httptest.NewServer(&handler)
|
|
defer server.Close()
|
|
client := clientset.NewForConfigOrDie(&restclient.Config{Host: server.URL, ContentConfig: restclient.ContentConfig{GroupVersion: &schema.GroupVersion{Group: "", Version: "v1"}}})
|
|
lw := NewListWatchFromClient(client.Core().RESTClient(), item.resource, item.namespace, item.fieldSelector)
|
|
// This test merely tests that the correct request is made.
|
|
lw.Watch(metav1.ListOptions{ResourceVersion: item.rv})
|
|
handler.ValidateRequest(t, item.location, "GET", nil)
|
|
}
|
|
}
|
|
|
|
type lw struct {
|
|
list runtime.Object
|
|
watch watch.Interface
|
|
}
|
|
|
|
func (w lw) List(options metav1.ListOptions) (runtime.Object, error) {
|
|
return w.list, nil
|
|
}
|
|
|
|
func (w lw) Watch(options metav1.ListOptions) (watch.Interface, error) {
|
|
return w.watch, nil
|
|
}
|
|
|
|
func TestListWatchUntil(t *testing.T) {
|
|
fw := watch.NewFake()
|
|
go func() {
|
|
var obj *v1.Pod
|
|
fw.Modify(obj)
|
|
}()
|
|
listwatch := lw{
|
|
list: &v1.PodList{Items: []v1.Pod{{}}},
|
|
watch: fw,
|
|
}
|
|
|
|
conditions := []watchtools.ConditionFunc{
|
|
func(event watch.Event) (bool, error) {
|
|
t.Logf("got %#v", event)
|
|
return event.Type == watch.Added, nil
|
|
},
|
|
func(event watch.Event) (bool, error) {
|
|
t.Logf("got %#v", event)
|
|
return event.Type == watch.Modified, nil
|
|
},
|
|
}
|
|
|
|
timeout := 10 * time.Second
|
|
lastEvent, err := watchtools.ListWatchUntil(timeout, listwatch, conditions...)
|
|
if err != nil {
|
|
t.Fatalf("expected nil error, got %#v", err)
|
|
}
|
|
if lastEvent == nil {
|
|
t.Fatal("expected an event")
|
|
}
|
|
if lastEvent.Type != watch.Modified {
|
|
t.Fatalf("expected MODIFIED event type, got %v", lastEvent.Type)
|
|
}
|
|
if got, isPod := lastEvent.Object.(*v1.Pod); !isPod {
|
|
t.Fatalf("expected a pod event, got %#v", got)
|
|
}
|
|
}
|