2014-08-30 00:58:25 +00:00
|
|
|
/*
|
2015-05-01 16:19:44 +00:00
|
|
|
Copyright 2014 The Kubernetes Authors All rights reserved.
|
2014-08-30 00:58:25 +00:00
|
|
|
|
|
|
|
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 api_test
|
|
|
|
|
|
|
|
import (
|
2014-09-19 01:27:33 +00:00
|
|
|
"encoding/json"
|
|
|
|
"math/rand"
|
2014-08-30 00:58:25 +00:00
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
|
2015-08-05 22:05:17 +00:00
|
|
|
"github.com/davecgh/go-spew/spew"
|
2015-11-26 23:45:43 +00:00
|
|
|
flag "github.com/spf13/pflag"
|
|
|
|
"github.com/ugorji/go/codec"
|
|
|
|
|
2015-08-05 22:03:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/api"
|
|
|
|
"k8s.io/kubernetes/pkg/api/meta"
|
|
|
|
"k8s.io/kubernetes/pkg/api/testapi"
|
|
|
|
apitesting "k8s.io/kubernetes/pkg/api/testing"
|
2015-10-27 13:29:18 +00:00
|
|
|
"k8s.io/kubernetes/pkg/api/unversioned"
|
2015-11-26 23:45:43 +00:00
|
|
|
"k8s.io/kubernetes/pkg/api/v1"
|
|
|
|
_ "k8s.io/kubernetes/pkg/apis/extensions"
|
|
|
|
_ "k8s.io/kubernetes/pkg/apis/extensions/v1beta1"
|
2015-08-05 22:03:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/runtime"
|
|
|
|
"k8s.io/kubernetes/pkg/util"
|
2015-09-09 17:45:01 +00:00
|
|
|
"k8s.io/kubernetes/pkg/util/sets"
|
2014-08-30 00:58:25 +00:00
|
|
|
)
|
|
|
|
|
2015-08-11 20:29:50 +00:00
|
|
|
var fuzzIters = flag.Int("fuzz-iters", 20, "How many fuzzing iterations to do.")
|
2014-12-12 21:57:25 +00:00
|
|
|
|
2015-12-17 21:20:22 +00:00
|
|
|
var codecsToTest = []func(version unversioned.GroupVersion, item runtime.Object) (runtime.Codec, error){
|
|
|
|
func(version unversioned.GroupVersion, item runtime.Object) (runtime.Codec, error) {
|
2015-11-26 23:45:43 +00:00
|
|
|
return testapi.GetCodecForObject(item)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2015-12-17 21:20:22 +00:00
|
|
|
func fuzzInternalObject(t *testing.T, forVersion unversioned.GroupVersion, item runtime.Object, seed int64) runtime.Object {
|
2015-01-29 00:57:33 +00:00
|
|
|
apitesting.FuzzerFor(t, forVersion, rand.NewSource(seed)).Fuzz(item)
|
2014-12-12 21:57:25 +00:00
|
|
|
|
2015-01-19 02:22:55 +00:00
|
|
|
j, err := meta.TypeAccessor(item)
|
2014-08-30 00:58:25 +00:00
|
|
|
if err != nil {
|
2014-12-12 21:57:25 +00:00
|
|
|
t.Fatalf("Unexpected error %v for %#v", err, item)
|
2014-08-30 00:58:25 +00:00
|
|
|
}
|
|
|
|
j.SetKind("")
|
|
|
|
j.SetAPIVersion("")
|
|
|
|
|
2014-12-12 21:57:25 +00:00
|
|
|
return item
|
|
|
|
}
|
|
|
|
|
|
|
|
func roundTrip(t *testing.T, codec runtime.Codec, item runtime.Object) {
|
2015-12-21 05:21:26 +00:00
|
|
|
t.Logf("codec: %#v", codec)
|
|
|
|
|
2015-03-06 04:55:32 +00:00
|
|
|
printer := spew.ConfigState{DisableMethods: true}
|
|
|
|
|
2014-12-12 21:57:25 +00:00
|
|
|
name := reflect.TypeOf(item).Elem().Name()
|
2015-12-10 02:15:02 +00:00
|
|
|
data, err := runtime.Encode(codec, item)
|
2014-08-30 00:58:25 +00:00
|
|
|
if err != nil {
|
2015-03-06 04:55:32 +00:00
|
|
|
t.Errorf("%v: %v (%s)", name, err, printer.Sprintf("%#v", item))
|
2014-08-30 00:58:25 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-12-10 02:15:02 +00:00
|
|
|
obj2, err := runtime.Decode(codec, data)
|
2014-08-30 00:58:25 +00:00
|
|
|
if err != nil {
|
2015-03-06 04:55:32 +00:00
|
|
|
t.Errorf("0: %v: %v\nCodec: %v\nData: %s\nSource: %#v", name, err, codec, string(data), printer.Sprintf("%#v", item))
|
2014-08-30 00:58:25 +00:00
|
|
|
return
|
|
|
|
}
|
2014-12-12 21:57:25 +00:00
|
|
|
if !api.Semantic.DeepEqual(item, obj2) {
|
2015-11-10 08:51:47 +00:00
|
|
|
t.Errorf("\n1: %v: diff: %v\nCodec: %v\nSource:\n\n%#v\n\nEncoded:\n\n%s\n\nFinal:\n\n%#v", name, util.ObjectGoPrintDiff(item, obj2), codec, printer.Sprintf("%#v", item), string(data), printer.Sprintf("%#v", obj2))
|
2014-11-13 12:01:25 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2014-12-12 21:57:25 +00:00
|
|
|
obj3 := reflect.New(reflect.TypeOf(item).Elem()).Interface().(runtime.Object)
|
2015-12-21 05:21:26 +00:00
|
|
|
if err := runtime.DecodeInto(codec, data, obj3); err != nil {
|
2014-08-30 00:58:25 +00:00
|
|
|
t.Errorf("2: %v: %v", name, err)
|
|
|
|
return
|
2014-11-13 12:01:25 +00:00
|
|
|
}
|
2014-12-12 21:57:25 +00:00
|
|
|
if !api.Semantic.DeepEqual(item, obj3) {
|
|
|
|
t.Errorf("3: %v: diff: %v\nCodec: %v", name, util.ObjectDiff(item, obj3), codec)
|
2014-11-13 12:01:25 +00:00
|
|
|
return
|
2014-08-30 00:58:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-12 21:57:25 +00:00
|
|
|
// roundTripSame verifies the same source object is tested in all API versions.
|
2015-03-04 00:54:17 +00:00
|
|
|
func roundTripSame(t *testing.T, item runtime.Object, except ...string) {
|
2015-09-09 17:45:01 +00:00
|
|
|
set := sets.NewString(except...)
|
2014-12-12 21:57:25 +00:00
|
|
|
seed := rand.Int63()
|
2015-12-17 21:20:22 +00:00
|
|
|
fuzzInternalObject(t, testapi.Default.InternalGroupVersion(), item, seed)
|
2015-09-17 05:15:05 +00:00
|
|
|
|
2015-12-17 21:20:22 +00:00
|
|
|
version := *testapi.Default.GroupVersion()
|
2015-11-26 23:45:43 +00:00
|
|
|
codecs := []runtime.Codec{}
|
|
|
|
for _, fn := range codecsToTest {
|
|
|
|
codec, err := fn(version, item)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("unable to get codec: %v", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
codecs = append(codecs, codec)
|
2015-09-17 05:15:05 +00:00
|
|
|
}
|
|
|
|
|
2015-12-17 21:20:22 +00:00
|
|
|
if !set.Has(version.String()) {
|
2015-06-30 02:30:14 +00:00
|
|
|
fuzzInternalObject(t, version, item, seed)
|
2015-11-26 23:45:43 +00:00
|
|
|
for _, codec := range codecs {
|
|
|
|
roundTrip(t, codec, item)
|
|
|
|
}
|
2015-06-03 17:31:26 +00:00
|
|
|
}
|
2014-12-12 21:57:25 +00:00
|
|
|
}
|
|
|
|
|
2014-10-05 21:17:25 +00:00
|
|
|
// For debugging problems
|
|
|
|
func TestSpecificKind(t *testing.T) {
|
2015-12-28 10:39:50 +00:00
|
|
|
// api.Scheme.Log(t)
|
|
|
|
// defer api.Scheme.Log(nil)
|
2014-12-12 21:57:25 +00:00
|
|
|
|
2015-12-21 05:21:26 +00:00
|
|
|
kind := "List"
|
2015-11-18 18:24:54 +00:00
|
|
|
for i := 0; i < *fuzzIters; i++ {
|
|
|
|
doRoundTripTest(kind, t)
|
|
|
|
if t.Failed() {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2014-10-05 21:17:25 +00:00
|
|
|
}
|
|
|
|
|
2014-12-08 02:19:10 +00:00
|
|
|
func TestList(t *testing.T) {
|
2015-12-28 10:39:50 +00:00
|
|
|
// api.Scheme.Log(t)
|
|
|
|
// defer api.Scheme.Log(nil)
|
2014-12-12 21:57:25 +00:00
|
|
|
|
2014-12-08 02:19:10 +00:00
|
|
|
kind := "List"
|
2015-12-08 19:40:23 +00:00
|
|
|
item, err := api.Scheme.New(api.SchemeGroupVersion.WithKind(kind))
|
2014-12-08 02:19:10 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Couldn't make a %v? %v", kind, err)
|
|
|
|
return
|
|
|
|
}
|
2014-12-12 21:57:25 +00:00
|
|
|
roundTripSame(t, item)
|
2014-12-08 02:19:10 +00:00
|
|
|
}
|
|
|
|
|
2015-12-01 23:45:29 +00:00
|
|
|
var nonRoundTrippableTypes = sets.NewString("ExportOptions")
|
2015-11-10 08:51:47 +00:00
|
|
|
|
2015-12-01 23:45:29 +00:00
|
|
|
var nonInternalRoundTrippableTypes = sets.NewString("List", "ListOptions", "ExportOptions")
|
2015-05-22 15:20:27 +00:00
|
|
|
var nonRoundTrippableTypesByVersion = map[string][]string{}
|
2014-11-07 02:09:46 +00:00
|
|
|
|
|
|
|
func TestRoundTripTypes(t *testing.T) {
|
2014-12-12 21:57:25 +00:00
|
|
|
// api.Scheme.Log(t)
|
|
|
|
// defer api.Scheme.Log(nil)
|
|
|
|
|
2015-11-18 15:34:16 +00:00
|
|
|
for kind := range api.Scheme.KnownTypes(testapi.Default.InternalGroupVersion()) {
|
2015-11-20 12:38:32 +00:00
|
|
|
t.Logf("working on %v in %v", kind, testapi.Default.InternalGroupVersion())
|
2014-11-07 02:09:46 +00:00
|
|
|
if nonRoundTrippableTypes.Has(kind) {
|
|
|
|
continue
|
|
|
|
}
|
2014-08-30 00:58:25 +00:00
|
|
|
// Try a few times, since runTest uses random values.
|
|
|
|
for i := 0; i < *fuzzIters; i++ {
|
2015-09-08 17:18:28 +00:00
|
|
|
doRoundTripTest(kind, t)
|
2015-11-18 18:24:54 +00:00
|
|
|
if t.Failed() {
|
|
|
|
break
|
|
|
|
}
|
2014-08-30 00:58:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-08 17:18:28 +00:00
|
|
|
func doRoundTripTest(kind string, t *testing.T) {
|
2015-12-08 19:40:23 +00:00
|
|
|
item, err := api.Scheme.New(testapi.Default.InternalGroupVersion().WithKind(kind))
|
2015-09-08 17:18:28 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Couldn't make a %v? %v", kind, err)
|
|
|
|
}
|
|
|
|
if _, err := meta.TypeAccessor(item); err != nil {
|
|
|
|
t.Fatalf("%q is not a TypeMeta and cannot be tested - add it to nonRoundTrippableTypes: %v", kind, err)
|
|
|
|
}
|
2015-11-20 12:38:32 +00:00
|
|
|
if api.Scheme.Recognizes(testapi.Default.GroupVersion().WithKind(kind)) {
|
2015-11-26 23:45:43 +00:00
|
|
|
roundTripSame(t, item, nonRoundTrippableTypesByVersion[kind]...)
|
|
|
|
}
|
2015-09-08 17:18:28 +00:00
|
|
|
if !nonInternalRoundTrippableTypes.Has(kind) {
|
2015-12-21 05:21:26 +00:00
|
|
|
roundTrip(t, testapi.Default.Codec(), fuzzInternalObject(t, testapi.Default.InternalGroupVersion(), item, rand.Int63()))
|
2015-09-08 17:18:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-30 00:58:25 +00:00
|
|
|
func TestEncode_Ptr(t *testing.T) {
|
2015-08-19 23:59:43 +00:00
|
|
|
grace := int64(30)
|
2014-08-30 00:58:25 +00:00
|
|
|
pod := &api.Pod{
|
2014-10-23 20:51:34 +00:00
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Labels: map[string]string{"name": "foo"},
|
|
|
|
},
|
2015-01-26 17:52:50 +00:00
|
|
|
Spec: api.PodSpec{
|
2015-03-14 01:38:07 +00:00
|
|
|
RestartPolicy: api.RestartPolicyAlways,
|
2015-01-26 17:52:50 +00:00
|
|
|
DNSPolicy: api.DNSClusterFirst,
|
2015-08-19 23:59:43 +00:00
|
|
|
|
|
|
|
TerminationGracePeriodSeconds: &grace,
|
2015-09-14 21:56:51 +00:00
|
|
|
|
|
|
|
SecurityContext: &api.PodSecurityContext{},
|
2015-01-26 17:52:50 +00:00
|
|
|
},
|
2014-08-30 00:58:25 +00:00
|
|
|
}
|
2014-09-06 01:16:28 +00:00
|
|
|
obj := runtime.Object(pod)
|
2015-12-21 05:21:26 +00:00
|
|
|
data, err := runtime.Encode(testapi.Default.Codec(), obj)
|
|
|
|
obj2, err2 := runtime.Decode(testapi.Default.Codec(), data)
|
2014-08-30 00:58:25 +00:00
|
|
|
if err != nil || err2 != nil {
|
|
|
|
t.Fatalf("Failure: '%v' '%v'", err, err2)
|
|
|
|
}
|
|
|
|
if _, ok := obj2.(*api.Pod); !ok {
|
|
|
|
t.Fatalf("Got wrong type")
|
|
|
|
}
|
2015-01-05 21:38:39 +00:00
|
|
|
if !api.Semantic.DeepEqual(obj2, pod) {
|
2015-09-14 21:56:51 +00:00
|
|
|
t.Errorf("\nExpected:\n\n %#v,\n\nGot:\n\n %#vDiff: %v\n\n", pod, obj2, util.ObjectDiff(obj2, pod))
|
|
|
|
|
2014-08-30 00:58:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestBadJSONRejection(t *testing.T) {
|
|
|
|
badJSONMissingKind := []byte(`{ }`)
|
2015-12-21 05:21:26 +00:00
|
|
|
if _, err := runtime.Decode(testapi.Default.Codec(), badJSONMissingKind); err == nil {
|
2014-08-30 00:58:25 +00:00
|
|
|
t.Errorf("Did not reject despite lack of kind field: %s", badJSONMissingKind)
|
|
|
|
}
|
|
|
|
badJSONUnknownType := []byte(`{"kind": "bar"}`)
|
2015-12-21 05:21:26 +00:00
|
|
|
if _, err1 := runtime.Decode(testapi.Default.Codec(), badJSONUnknownType); err1 == nil {
|
2014-08-30 00:58:25 +00:00
|
|
|
t.Errorf("Did not reject despite use of unknown type: %s", badJSONUnknownType)
|
|
|
|
}
|
|
|
|
/*badJSONKindMismatch := []byte(`{"kind": "Pod"}`)
|
|
|
|
if err2 := DecodeInto(badJSONKindMismatch, &Minion{}); err2 == nil {
|
|
|
|
t.Errorf("Kind is set but doesn't match the object type: %s", badJSONKindMismatch)
|
|
|
|
}*/
|
|
|
|
}
|
2014-09-19 01:27:33 +00:00
|
|
|
|
2015-10-27 13:29:18 +00:00
|
|
|
func TestUnversionedTypes(t *testing.T) {
|
|
|
|
testcases := []runtime.Object{
|
|
|
|
&unversioned.Status{Status: "Failure", Message: "something went wrong"},
|
|
|
|
&unversioned.APIVersions{Versions: []string{"A", "B", "C"}},
|
|
|
|
&unversioned.APIGroupList{Groups: []unversioned.APIGroup{{Name: "mygroup"}}},
|
|
|
|
&unversioned.APIGroup{Name: "mygroup"},
|
|
|
|
&unversioned.APIResourceList{GroupVersion: "mygroup/myversion"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, obj := range testcases {
|
|
|
|
// Make sure the unversioned codec can encode
|
2015-12-21 05:21:26 +00:00
|
|
|
unversionedJSON, err := runtime.Encode(testapi.Default.Codec(), obj)
|
2015-10-27 13:29:18 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("%v: unexpected error: %v", obj, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure the versioned codec under test can decode
|
2015-12-21 05:21:26 +00:00
|
|
|
versionDecodedObject, err := runtime.Decode(testapi.Default.Codec(), unversionedJSON)
|
2015-10-27 13:29:18 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("%v: unexpected error: %v", obj, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// Make sure it decodes correctly
|
|
|
|
if !reflect.DeepEqual(obj, versionDecodedObject) {
|
|
|
|
t.Errorf("%v: expected %#v, got %#v", obj, obj, versionDecodedObject)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-19 01:27:33 +00:00
|
|
|
const benchmarkSeed = 100
|
|
|
|
|
2015-11-26 23:45:43 +00:00
|
|
|
func benchmarkItems() []v1.Pod {
|
2015-12-17 21:20:22 +00:00
|
|
|
apiObjectFuzzer := apitesting.FuzzerFor(nil, api.SchemeGroupVersion, rand.NewSource(benchmarkSeed))
|
2015-11-26 23:45:43 +00:00
|
|
|
items := make([]v1.Pod, 2)
|
|
|
|
for i := range items {
|
|
|
|
apiObjectFuzzer.Fuzz(&items[i])
|
|
|
|
}
|
|
|
|
return items
|
|
|
|
}
|
|
|
|
|
|
|
|
// BenchmarkEncodeCodec measures the cost of performing a codec encode, which includes
|
|
|
|
// reflection (to clear APIVersion and Kind)
|
|
|
|
func BenchmarkEncodeCodec(b *testing.B) {
|
|
|
|
items := benchmarkItems()
|
|
|
|
width := len(items)
|
|
|
|
b.ResetTimer()
|
2014-09-19 01:27:33 +00:00
|
|
|
for i := 0; i < b.N; i++ {
|
2015-12-21 05:21:26 +00:00
|
|
|
if _, err := runtime.Encode(testapi.Default.Codec(), &items[i%width]); err != nil {
|
2015-11-26 23:45:43 +00:00
|
|
|
b.Fatal(err)
|
|
|
|
}
|
2014-09-19 01:27:33 +00:00
|
|
|
}
|
2015-11-26 23:45:43 +00:00
|
|
|
b.StopTimer()
|
2014-09-19 01:27:33 +00:00
|
|
|
}
|
|
|
|
|
2015-11-26 23:45:43 +00:00
|
|
|
// BenchmarkEncodeJSONMarshal provides a baseline for regular JSON encode performance
|
|
|
|
func BenchmarkEncodeJSONMarshal(b *testing.B) {
|
|
|
|
items := benchmarkItems()
|
|
|
|
width := len(items)
|
|
|
|
b.ResetTimer()
|
2014-09-19 01:27:33 +00:00
|
|
|
for i := 0; i < b.N; i++ {
|
2015-11-26 23:45:43 +00:00
|
|
|
if _, err := json.Marshal(&items[i%width]); err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
2014-09-19 01:27:33 +00:00
|
|
|
}
|
2015-11-26 23:45:43 +00:00
|
|
|
b.StopTimer()
|
2014-09-19 01:27:33 +00:00
|
|
|
}
|
|
|
|
|
2015-11-26 23:45:43 +00:00
|
|
|
func BenchmarkDecodeCodec(b *testing.B) {
|
|
|
|
codec := testapi.Default.Codec()
|
|
|
|
items := benchmarkItems()
|
|
|
|
width := len(items)
|
|
|
|
encoded := make([][]byte, width)
|
|
|
|
for i := range items {
|
2015-12-10 02:15:02 +00:00
|
|
|
data, err := runtime.Encode(codec, &items[i])
|
2015-11-26 23:45:43 +00:00
|
|
|
if err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
encoded[i] = data
|
|
|
|
}
|
|
|
|
|
|
|
|
b.ResetTimer()
|
2014-09-19 01:27:33 +00:00
|
|
|
for i := 0; i < b.N; i++ {
|
2015-12-10 02:15:02 +00:00
|
|
|
if _, err := runtime.Decode(codec, encoded[i%width]); err != nil {
|
2015-11-26 23:45:43 +00:00
|
|
|
b.Fatal(err)
|
|
|
|
}
|
2014-09-19 01:27:33 +00:00
|
|
|
}
|
2015-11-26 23:45:43 +00:00
|
|
|
b.StopTimer()
|
2014-09-19 01:27:33 +00:00
|
|
|
}
|
|
|
|
|
2015-12-21 05:21:26 +00:00
|
|
|
func BenchmarkDecodeIntoExternalCodec(b *testing.B) {
|
2015-11-26 23:45:43 +00:00
|
|
|
codec := testapi.Default.Codec()
|
|
|
|
items := benchmarkItems()
|
|
|
|
width := len(items)
|
|
|
|
encoded := make([][]byte, width)
|
|
|
|
for i := range items {
|
2015-12-10 02:15:02 +00:00
|
|
|
data, err := runtime.Encode(codec, &items[i])
|
2015-11-26 23:45:43 +00:00
|
|
|
if err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
encoded[i] = data
|
|
|
|
}
|
|
|
|
|
|
|
|
b.ResetTimer()
|
2014-09-19 01:27:33 +00:00
|
|
|
for i := 0; i < b.N; i++ {
|
2015-11-26 23:45:43 +00:00
|
|
|
obj := v1.Pod{}
|
2015-12-10 02:15:02 +00:00
|
|
|
if err := runtime.DecodeInto(codec, encoded[i%width], &obj); err != nil {
|
2015-11-26 23:45:43 +00:00
|
|
|
b.Fatal(err)
|
|
|
|
}
|
2014-09-19 01:27:33 +00:00
|
|
|
}
|
2015-11-26 23:45:43 +00:00
|
|
|
b.StopTimer()
|
2014-09-19 01:27:33 +00:00
|
|
|
}
|
|
|
|
|
2015-12-21 05:21:26 +00:00
|
|
|
func BenchmarkDecodeIntoInternalCodec(b *testing.B) {
|
|
|
|
codec := testapi.Default.Codec()
|
|
|
|
items := benchmarkItems()
|
|
|
|
width := len(items)
|
|
|
|
encoded := make([][]byte, width)
|
|
|
|
for i := range items {
|
|
|
|
data, err := runtime.Encode(codec, &items[i])
|
|
|
|
if err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
encoded[i] = data
|
|
|
|
}
|
|
|
|
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
obj := api.Pod{}
|
|
|
|
if err := runtime.DecodeInto(codec, encoded[i%width], &obj); err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
b.StopTimer()
|
|
|
|
}
|
|
|
|
|
2014-09-19 01:27:33 +00:00
|
|
|
// BenchmarkDecodeJSON provides a baseline for regular JSON decode performance
|
2015-11-26 23:45:43 +00:00
|
|
|
func BenchmarkDecodeIntoJSON(b *testing.B) {
|
|
|
|
codec := testapi.Default.Codec()
|
|
|
|
items := benchmarkItems()
|
|
|
|
width := len(items)
|
|
|
|
encoded := make([][]byte, width)
|
|
|
|
for i := range items {
|
2015-12-10 02:15:02 +00:00
|
|
|
data, err := runtime.Encode(codec, &items[i])
|
2015-11-26 23:45:43 +00:00
|
|
|
if err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
encoded[i] = data
|
|
|
|
}
|
|
|
|
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
obj := v1.Pod{}
|
|
|
|
if err := json.Unmarshal(encoded[i%width], &obj); err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
b.StopTimer()
|
|
|
|
}
|
|
|
|
|
|
|
|
// BenchmarkDecodeJSON provides a baseline for codecgen JSON decode performance
|
|
|
|
func BenchmarkDecodeIntoJSONCodecGen(b *testing.B) {
|
|
|
|
kcodec := testapi.Default.Codec()
|
|
|
|
items := benchmarkItems()
|
|
|
|
width := len(items)
|
|
|
|
encoded := make([][]byte, width)
|
|
|
|
for i := range items {
|
2015-12-10 02:15:02 +00:00
|
|
|
data, err := runtime.Encode(kcodec, &items[i])
|
2015-11-26 23:45:43 +00:00
|
|
|
if err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
encoded[i] = data
|
|
|
|
}
|
|
|
|
handler := &codec.JsonHandle{}
|
|
|
|
|
|
|
|
b.ResetTimer()
|
2014-09-19 01:27:33 +00:00
|
|
|
for i := 0; i < b.N; i++ {
|
2015-11-26 23:45:43 +00:00
|
|
|
obj := v1.Pod{}
|
|
|
|
if err := codec.NewDecoderBytes(encoded[i%width], handler).Decode(&obj); err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
2014-09-19 01:27:33 +00:00
|
|
|
}
|
2015-11-26 23:45:43 +00:00
|
|
|
b.StopTimer()
|
2014-09-19 01:27:33 +00:00
|
|
|
}
|