| /* |
| 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 storage |
| |
| import ( |
| "testing" |
| |
| apiequality "k8s.io/apimachinery/pkg/api/equality" |
| "k8s.io/apimachinery/pkg/api/resource" |
| metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" |
| "k8s.io/apimachinery/pkg/fields" |
| "k8s.io/apimachinery/pkg/labels" |
| "k8s.io/apimachinery/pkg/util/diff" |
| genericapirequest "k8s.io/apiserver/pkg/endpoints/request" |
| "k8s.io/apiserver/pkg/registry/generic" |
| genericregistrytest "k8s.io/apiserver/pkg/registry/generic/testing" |
| "k8s.io/apiserver/pkg/registry/rest" |
| etcdtesting "k8s.io/apiserver/pkg/storage/etcd/testing" |
| api "k8s.io/kubernetes/pkg/apis/core" |
| "k8s.io/kubernetes/pkg/registry/registrytest" |
| ) |
| |
| func newStorage(t *testing.T) (*REST, *StatusREST, *etcdtesting.EtcdTestServer) { |
| etcdStorage, server := registrytest.NewEtcdStorage(t, "") |
| restOptions := generic.RESTOptions{ |
| StorageConfig: etcdStorage, |
| Decorator: generic.UndecoratedStorage, |
| DeleteCollectionWorkers: 1, |
| ResourcePrefix: "resourcequotas", |
| } |
| resourceQuotaStorage, statusStorage := NewREST(restOptions) |
| return resourceQuotaStorage, statusStorage, server |
| } |
| |
| func validNewResourceQuota() *api.ResourceQuota { |
| return &api.ResourceQuota{ |
| ObjectMeta: metav1.ObjectMeta{ |
| Name: "foo", |
| Namespace: metav1.NamespaceDefault, |
| }, |
| Spec: api.ResourceQuotaSpec{ |
| Hard: api.ResourceList{ |
| api.ResourceCPU: resource.MustParse("100"), |
| api.ResourceMemory: resource.MustParse("4Gi"), |
| api.ResourcePods: resource.MustParse("10"), |
| api.ResourceServices: resource.MustParse("10"), |
| api.ResourceReplicationControllers: resource.MustParse("10"), |
| api.ResourceQuotas: resource.MustParse("1"), |
| }, |
| }, |
| } |
| } |
| |
| func TestCreate(t *testing.T) { |
| storage, _, server := newStorage(t) |
| defer server.Terminate(t) |
| defer storage.Store.DestroyFunc() |
| test := genericregistrytest.New(t, storage.Store) |
| resourcequota := validNewResourceQuota() |
| resourcequota.ObjectMeta = metav1.ObjectMeta{} |
| test.TestCreate( |
| // valid |
| resourcequota, |
| // invalid |
| &api.ResourceQuota{ |
| ObjectMeta: metav1.ObjectMeta{Name: "_-a123-a_"}, |
| }, |
| ) |
| } |
| |
| func TestCreateSetsFields(t *testing.T) { |
| storage, _, server := newStorage(t) |
| defer server.Terminate(t) |
| defer storage.Store.DestroyFunc() |
| ctx := genericapirequest.NewDefaultContext() |
| resourcequota := validNewResourceQuota() |
| _, err := storage.Create(genericapirequest.NewDefaultContext(), resourcequota, rest.ValidateAllObjectFunc, &metav1.CreateOptions{}) |
| if err != nil { |
| t.Fatalf("unexpected error: %v", err) |
| } |
| |
| object, err := storage.Get(ctx, "foo", &metav1.GetOptions{}) |
| if err != nil { |
| t.Errorf("unexpected error: %v", err) |
| } |
| actual := object.(*api.ResourceQuota) |
| if actual.Name != resourcequota.Name { |
| t.Errorf("unexpected resourcequota: %#v", actual) |
| } |
| if len(actual.UID) == 0 { |
| t.Errorf("expected resourcequota UID to be set: %#v", actual) |
| } |
| } |
| |
| func TestDelete(t *testing.T) { |
| storage, _, server := newStorage(t) |
| defer server.Terminate(t) |
| defer storage.Store.DestroyFunc() |
| test := genericregistrytest.New(t, storage.Store).ReturnDeletedObject() |
| test.TestDelete(validNewResourceQuota()) |
| } |
| |
| func TestGet(t *testing.T) { |
| storage, _, server := newStorage(t) |
| defer server.Terminate(t) |
| defer storage.Store.DestroyFunc() |
| test := genericregistrytest.New(t, storage.Store) |
| test.TestGet(validNewResourceQuota()) |
| } |
| |
| func TestList(t *testing.T) { |
| storage, _, server := newStorage(t) |
| defer server.Terminate(t) |
| defer storage.Store.DestroyFunc() |
| test := genericregistrytest.New(t, storage.Store) |
| test.TestList(validNewResourceQuota()) |
| } |
| |
| func TestWatch(t *testing.T) { |
| storage, _, server := newStorage(t) |
| defer server.Terminate(t) |
| defer storage.Store.DestroyFunc() |
| test := genericregistrytest.New(t, storage.Store) |
| test.TestWatch( |
| validNewResourceQuota(), |
| // matching labels |
| []labels.Set{}, |
| // not matching labels |
| []labels.Set{ |
| {"foo": "bar"}, |
| }, |
| // matching fields |
| []fields.Set{ |
| {"metadata.name": "foo"}, |
| }, |
| // not matching fields |
| []fields.Set{ |
| {"metadata.name": "bar"}, |
| }, |
| ) |
| } |
| |
| func TestUpdateStatus(t *testing.T) { |
| storage, status, server := newStorage(t) |
| defer server.Terminate(t) |
| defer storage.Store.DestroyFunc() |
| ctx := genericapirequest.NewDefaultContext() |
| |
| key, _ := storage.KeyFunc(ctx, "foo") |
| resourcequotaStart := validNewResourceQuota() |
| err := storage.Storage.Create(ctx, key, resourcequotaStart, nil, 0, false) |
| if err != nil { |
| t.Fatalf("Unexpected error: %v", err) |
| } |
| |
| resourcequotaIn := &api.ResourceQuota{ |
| ObjectMeta: metav1.ObjectMeta{ |
| Name: "foo", |
| Namespace: metav1.NamespaceDefault, |
| }, |
| Status: api.ResourceQuotaStatus{ |
| Used: api.ResourceList{ |
| api.ResourceCPU: resource.MustParse("1"), |
| api.ResourceMemory: resource.MustParse("1Gi"), |
| api.ResourcePods: resource.MustParse("1"), |
| api.ResourceServices: resource.MustParse("1"), |
| api.ResourceReplicationControllers: resource.MustParse("1"), |
| api.ResourceQuotas: resource.MustParse("1"), |
| }, |
| Hard: api.ResourceList{ |
| api.ResourceCPU: resource.MustParse("100"), |
| api.ResourceMemory: resource.MustParse("4Gi"), |
| api.ResourcePods: resource.MustParse("10"), |
| api.ResourceServices: resource.MustParse("10"), |
| api.ResourceReplicationControllers: resource.MustParse("10"), |
| api.ResourceQuotas: resource.MustParse("1"), |
| }, |
| }, |
| } |
| |
| _, _, err = status.Update(ctx, resourcequotaIn.Name, rest.DefaultUpdatedObjectInfo(resourcequotaIn), rest.ValidateAllObjectFunc, rest.ValidateAllObjectUpdateFunc, false, &metav1.UpdateOptions{}) |
| if err != nil { |
| t.Fatalf("Unexpected error: %v", err) |
| } |
| obj, err := storage.Get(ctx, "foo", &metav1.GetOptions{}) |
| rqOut := obj.(*api.ResourceQuota) |
| // only compare the meaningful update b/c we can't compare due to metadata |
| if !apiequality.Semantic.DeepEqual(resourcequotaIn.Status, rqOut.Status) { |
| t.Errorf("unexpected object: %s", diff.ObjectDiff(resourcequotaIn, rqOut)) |
| } |
| } |
| |
| func TestShortNames(t *testing.T) { |
| storage, _, server := newStorage(t) |
| defer server.Terminate(t) |
| defer storage.Store.DestroyFunc() |
| expected := []string{"quota"} |
| registrytest.AssertShortNames(t, storage, expected) |
| } |