mirror of
https://github.com/optim-enterprises-bv/kubernetes.git
synced 2025-11-02 03:08:15 +00:00
The pending codec -> conversion split changes the signature of Encode and Decode to be more complicated. Create a stub helper with the exact semantics of today and do the simple mechanical refactor here to reduce the cost of that change.
263 lines
8.5 KiB
Go
263 lines
8.5 KiB
Go
/*
|
|
Copyright 2014 The Kubernetes Authors All rights reserved.
|
|
|
|
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 runtime_test
|
|
|
|
import (
|
|
"encoding/json"
|
|
"reflect"
|
|
"testing"
|
|
|
|
"k8s.io/kubernetes/pkg/api"
|
|
"k8s.io/kubernetes/pkg/api/meta"
|
|
"k8s.io/kubernetes/pkg/api/unversioned"
|
|
"k8s.io/kubernetes/pkg/runtime"
|
|
"k8s.io/kubernetes/pkg/util"
|
|
)
|
|
|
|
type EmbeddedTest struct {
|
|
runtime.TypeMeta
|
|
ID string
|
|
Object runtime.EmbeddedObject
|
|
EmptyObject runtime.EmbeddedObject
|
|
}
|
|
|
|
type EmbeddedTestExternal struct {
|
|
runtime.TypeMeta `json:",inline"`
|
|
ID string `json:"id,omitempty"`
|
|
Object runtime.RawExtension `json:"object,omitempty"`
|
|
EmptyObject runtime.RawExtension `json:"emptyObject,omitempty"`
|
|
}
|
|
|
|
type ObjectTest struct {
|
|
runtime.TypeMeta
|
|
|
|
ID string
|
|
Items []runtime.Object
|
|
}
|
|
|
|
type ObjectTestExternal struct {
|
|
runtime.TypeMeta `yaml:",inline" json:",inline"`
|
|
|
|
ID string `json:"id,omitempty"`
|
|
Items []runtime.RawExtension `json:"items,omitempty"`
|
|
}
|
|
|
|
func (*ObjectTest) IsAnAPIObject() {}
|
|
func (*ObjectTestExternal) IsAnAPIObject() {}
|
|
func (*EmbeddedTest) IsAnAPIObject() {}
|
|
func (*EmbeddedTestExternal) IsAnAPIObject() {}
|
|
|
|
func TestDecodeEmptyRawExtensionAsObject(t *testing.T) {
|
|
internalGV := unversioned.GroupVersion{Group: "test.group", Version: ""}
|
|
externalGV := unversioned.GroupVersion{Group: "test.group", Version: "v1test"}
|
|
externalGVK := externalGV.WithKind("ObjectTest")
|
|
|
|
s := runtime.NewScheme()
|
|
s.AddInternalGroupVersion(internalGV)
|
|
s.AddKnownTypes(internalGV, &ObjectTest{})
|
|
s.AddKnownTypeWithName(externalGVK, &ObjectTestExternal{})
|
|
|
|
obj, err := s.Decode([]byte(`{"kind":"` + externalGVK.Kind + `","apiVersion":"` + externalGV.String() + `","items":[{}]}`))
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
test := obj.(*ObjectTest)
|
|
if unk, ok := test.Items[0].(*runtime.Unknown); !ok || unk.Kind != "" || unk.APIVersion != "" || string(unk.RawJSON) != "{}" {
|
|
t.Fatalf("unexpected object: %#v", test.Items[0])
|
|
}
|
|
|
|
obj, err = s.Decode([]byte(`{"kind":"` + externalGVK.Kind + `","apiVersion":"` + externalGV.String() + `","items":[{"kind":"Other","apiVersion":"v1"}]}`))
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
test = obj.(*ObjectTest)
|
|
if unk, ok := test.Items[0].(*runtime.Unknown); !ok || unk.Kind != "Other" || unk.APIVersion != "v1" || string(unk.RawJSON) != `{"kind":"Other","apiVersion":"v1"}` {
|
|
t.Fatalf("unexpected object: %#v", test.Items[0])
|
|
}
|
|
}
|
|
|
|
func TestArrayOfRuntimeObject(t *testing.T) {
|
|
internalGV := unversioned.GroupVersion{Group: "test.group", Version: ""}
|
|
externalGV := unversioned.GroupVersion{Group: "test.group", Version: "v1test"}
|
|
|
|
s := runtime.NewScheme()
|
|
s.AddInternalGroupVersion(internalGV)
|
|
s.AddKnownTypes(internalGV, &EmbeddedTest{})
|
|
s.AddKnownTypeWithName(externalGV.WithKind("EmbeddedTest"), &EmbeddedTestExternal{})
|
|
s.AddKnownTypes(internalGV, &ObjectTest{})
|
|
s.AddKnownTypeWithName(externalGV.WithKind("ObjectTest"), &ObjectTestExternal{})
|
|
|
|
internal := &ObjectTest{
|
|
Items: []runtime.Object{
|
|
&EmbeddedTest{ID: "foo"},
|
|
&EmbeddedTest{ID: "bar"},
|
|
// TODO: until YAML is removed, this JSON must be in ascending key order to ensure consistent roundtrip serialization
|
|
&runtime.Unknown{RawJSON: []byte(`{"apiVersion":"unknown.group/unknown","foo":"bar","kind":"OtherTest"}`)},
|
|
&ObjectTest{
|
|
Items: []runtime.Object{
|
|
&EmbeddedTest{ID: "baz"},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
wire, err := s.EncodeToVersion(internal, externalGV.String())
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
t.Logf("Wire format is:\n%s\n", string(wire))
|
|
|
|
obj := &ObjectTestExternal{}
|
|
if err := json.Unmarshal(wire, obj); err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
t.Logf("exact wire is: %s", string(obj.Items[0].RawJSON))
|
|
|
|
decoded, err := runtime.Decode(s, wire)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
list, err := meta.ExtractList(decoded)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if errs := runtime.DecodeList(list, s); len(errs) > 0 {
|
|
t.Fatalf("unexpected error: %v", errs)
|
|
}
|
|
|
|
list2, err := meta.ExtractList(list[3])
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if errs := runtime.DecodeList(list2, s); len(errs) > 0 {
|
|
t.Fatalf("unexpected error: %v", errs)
|
|
}
|
|
if err := meta.SetList(list[3], list2); err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
|
|
internal.Items[2].(*runtime.Unknown).Kind = "OtherTest"
|
|
internal.Items[2].(*runtime.Unknown).APIVersion = "unknown.group/unknown"
|
|
if e, a := internal.Items, list; !reflect.DeepEqual(e, a) {
|
|
t.Errorf("mismatched decoded: %s", util.ObjectDiff(e, a))
|
|
}
|
|
}
|
|
|
|
func TestEmbeddedObject(t *testing.T) {
|
|
internalGV := unversioned.GroupVersion{Group: "test.group", Version: ""}
|
|
externalGV := unversioned.GroupVersion{Group: "test.group", Version: "v1test"}
|
|
embeddedTestExternalGVK := externalGV.WithKind("EmbeddedTest")
|
|
|
|
s := runtime.NewScheme()
|
|
s.AddInternalGroupVersion(internalGV)
|
|
s.AddKnownTypes(internalGV, &EmbeddedTest{})
|
|
s.AddKnownTypeWithName(embeddedTestExternalGVK, &EmbeddedTestExternal{})
|
|
|
|
outer := &EmbeddedTest{
|
|
ID: "outer",
|
|
Object: runtime.EmbeddedObject{
|
|
Object: &EmbeddedTest{
|
|
ID: "inner",
|
|
},
|
|
},
|
|
}
|
|
|
|
wire, err := s.EncodeToVersion(outer, externalGV.String())
|
|
if err != nil {
|
|
t.Fatalf("Unexpected encode error '%v'", err)
|
|
}
|
|
|
|
t.Logf("Wire format is:\n%v\n", string(wire))
|
|
|
|
decoded, err := runtime.Decode(s, wire)
|
|
if err != nil {
|
|
t.Fatalf("Unexpected decode error %v", err)
|
|
}
|
|
|
|
if e, a := outer, decoded; !reflect.DeepEqual(e, a) {
|
|
t.Errorf("Expected: %#v but got %#v", e, a)
|
|
}
|
|
|
|
// test JSON decoding of the external object, which should preserve
|
|
// raw bytes
|
|
var externalViaJSON EmbeddedTestExternal
|
|
err = json.Unmarshal(wire, &externalViaJSON)
|
|
if err != nil {
|
|
t.Fatalf("Unexpected decode error %v", err)
|
|
}
|
|
if externalViaJSON.Kind == "" || externalViaJSON.APIVersion == "" || externalViaJSON.ID != "outer" {
|
|
t.Errorf("Expected objects to have type info set, got %#v", externalViaJSON)
|
|
}
|
|
if !reflect.DeepEqual(externalViaJSON.EmptyObject.RawJSON, []byte("null")) || len(externalViaJSON.Object.RawJSON) == 0 {
|
|
t.Errorf("Expected deserialization of nested objects into bytes, got %#v", externalViaJSON)
|
|
}
|
|
|
|
// test JSON decoding, too, since Decode uses yaml unmarshalling.
|
|
// Generic Unmarshalling of JSON cannot load the nested objects because there is
|
|
// no default schema set. Consumers wishing to get direct JSON decoding must use
|
|
// the external representation
|
|
var decodedViaJSON EmbeddedTest
|
|
err = json.Unmarshal(wire, &decodedViaJSON)
|
|
if err != nil {
|
|
t.Fatalf("Unexpected decode error %v", err)
|
|
}
|
|
if a := decodedViaJSON; a.Object.Object != nil || a.EmptyObject.Object != nil {
|
|
t.Errorf("Expected embedded objects to be nil: %#v", a)
|
|
}
|
|
}
|
|
|
|
// TestDeepCopyOfEmbeddedObject checks to make sure that EmbeddedObject's can be passed through DeepCopy with fidelity
|
|
func TestDeepCopyOfEmbeddedObject(t *testing.T) {
|
|
internalGV := unversioned.GroupVersion{Group: "test.group", Version: ""}
|
|
externalGV := unversioned.GroupVersion{Group: "test.group", Version: "v1test"}
|
|
embeddedTestExternalGVK := externalGV.WithKind("EmbeddedTest")
|
|
|
|
s := runtime.NewScheme()
|
|
s.AddInternalGroupVersion(internalGV)
|
|
s.AddKnownTypes(internalGV, &EmbeddedTest{})
|
|
s.AddKnownTypeWithName(embeddedTestExternalGVK, &EmbeddedTestExternal{})
|
|
|
|
original := &EmbeddedTest{
|
|
ID: "outer",
|
|
Object: runtime.EmbeddedObject{
|
|
Object: &EmbeddedTest{
|
|
ID: "inner",
|
|
},
|
|
},
|
|
}
|
|
|
|
originalData, err := s.EncodeToVersion(original, externalGV.String())
|
|
if err != nil {
|
|
t.Errorf("unexpected error: %v", err)
|
|
}
|
|
t.Logf("originalRole = %v\n", string(originalData))
|
|
|
|
copyOfOriginal, err := api.Scheme.DeepCopy(original)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
|
|
copiedData, err := s.EncodeToVersion(copyOfOriginal.(runtime.Object), externalGV.String())
|
|
if err != nil {
|
|
t.Errorf("unexpected error: %v", err)
|
|
}
|
|
t.Logf("copyOfRole = %v\n", string(copiedData))
|
|
|
|
if !reflect.DeepEqual(original, copyOfOriginal) {
|
|
t.Errorf("expected \n%v\n, got \n%v", string(originalData), string(copiedData))
|
|
}
|
|
}
|