mirror of https://github.com/k3s-io/k3s
368 lines
11 KiB
Go
368 lines
11 KiB
Go
/*
|
|
Copyright 2014 Google Inc. 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 api_test
|
|
|
|
import (
|
|
"encoding/json"
|
|
|
|
"math/rand"
|
|
"reflect"
|
|
"strconv"
|
|
"testing"
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api/latest"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api/meta"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api/resource"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api/v1beta1"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api/v1beta2"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api/v1beta3"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/runtime"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/util"
|
|
|
|
docker "github.com/fsouza/go-dockerclient"
|
|
fuzz "github.com/google/gofuzz"
|
|
flag "github.com/spf13/pflag"
|
|
"speter.net/go/exp/math/dec/inf"
|
|
)
|
|
|
|
var fuzzIters = flag.Int("fuzz_iters", 20, "How many fuzzing iterations to do.")
|
|
|
|
// fuzzerFor can randomly populate api objects that are destined for version.
|
|
func fuzzerFor(t *testing.T, version string, src rand.Source) *fuzz.Fuzzer {
|
|
f := fuzz.New().NilChance(.5).NumElements(1, 1)
|
|
if src != nil {
|
|
f.RandSource(src)
|
|
}
|
|
f.Funcs(
|
|
func(j *runtime.PluginBase, c fuzz.Continue) {
|
|
// Do nothing; this struct has only a Kind field and it must stay blank in memory.
|
|
},
|
|
func(j *runtime.TypeMeta, c fuzz.Continue) {
|
|
// We have to customize the randomization of TypeMetas because their
|
|
// APIVersion and Kind must remain blank in memory.
|
|
j.APIVersion = ""
|
|
j.Kind = ""
|
|
},
|
|
func(j *api.TypeMeta, c fuzz.Continue) {
|
|
// We have to customize the randomization of TypeMetas because their
|
|
// APIVersion and Kind must remain blank in memory.
|
|
j.APIVersion = ""
|
|
j.Kind = ""
|
|
},
|
|
func(j *api.ObjectMeta, c fuzz.Continue) {
|
|
j.Name = c.RandString()
|
|
j.ResourceVersion = strconv.FormatUint(c.RandUint64(), 10)
|
|
j.SelfLink = c.RandString()
|
|
|
|
var sec, nsec int64
|
|
c.Fuzz(&sec)
|
|
c.Fuzz(&nsec)
|
|
j.CreationTimestamp = util.Unix(sec, nsec).Rfc3339Copy()
|
|
},
|
|
func(j *api.ListMeta, c fuzz.Continue) {
|
|
j.ResourceVersion = strconv.FormatUint(c.RandUint64(), 10)
|
|
j.SelfLink = c.RandString()
|
|
},
|
|
func(j *api.PodPhase, c fuzz.Continue) {
|
|
statuses := []api.PodPhase{api.PodPending, api.PodRunning, api.PodFailed, api.PodUnknown}
|
|
*j = statuses[c.Rand.Intn(len(statuses))]
|
|
},
|
|
func(j *api.ReplicationControllerSpec, c fuzz.Continue) {
|
|
// TemplateRef must be nil for round trip
|
|
c.Fuzz(&j.Template)
|
|
if j.Template == nil {
|
|
// TODO: v1beta1/2 can't round trip a nil template correctly, fix by having v1beta1/2
|
|
// conversion compare converted object to nil via DeepEqual
|
|
j.Template = &api.PodTemplateSpec{}
|
|
}
|
|
j.Template.ObjectMeta = api.ObjectMeta{Labels: j.Template.ObjectMeta.Labels}
|
|
c.Fuzz(&j.Selector)
|
|
j.Replicas = int(c.RandUint64())
|
|
},
|
|
func(j *api.ReplicationControllerStatus, c fuzz.Continue) {
|
|
// only replicas round trips
|
|
j.Replicas = int(c.RandUint64())
|
|
},
|
|
func(j *api.List, c fuzz.Continue) {
|
|
c.Fuzz(&j.ListMeta)
|
|
c.Fuzz(&j.Items)
|
|
if j.Items == nil {
|
|
j.Items = []runtime.Object{}
|
|
}
|
|
},
|
|
func(j *runtime.Object, c fuzz.Continue) {
|
|
if c.RandBool() {
|
|
*j = &runtime.Unknown{
|
|
TypeMeta: runtime.TypeMeta{Kind: "Something", APIVersion: "unknown"},
|
|
RawJSON: []byte(`{"apiVersion":"unknown","kind":"Something","someKey":"someValue"}`),
|
|
}
|
|
} else {
|
|
types := []runtime.Object{&api.Pod{}, &api.ReplicationController{}}
|
|
t := types[c.Rand.Intn(len(types))]
|
|
c.Fuzz(t)
|
|
*j = t
|
|
}
|
|
},
|
|
func(intstr *util.IntOrString, c fuzz.Continue) {
|
|
// util.IntOrString will panic if its kind is set wrong.
|
|
if c.RandBool() {
|
|
intstr.Kind = util.IntstrInt
|
|
intstr.IntVal = int(c.RandUint64())
|
|
intstr.StrVal = ""
|
|
} else {
|
|
intstr.Kind = util.IntstrString
|
|
intstr.IntVal = 0
|
|
intstr.StrVal = c.RandString()
|
|
}
|
|
},
|
|
func(pb map[docker.Port][]docker.PortBinding, c fuzz.Continue) {
|
|
// This is necessary because keys with nil values get omitted.
|
|
// TODO: Is this a bug?
|
|
pb[docker.Port(c.RandString())] = []docker.PortBinding{
|
|
{c.RandString(), c.RandString()},
|
|
{c.RandString(), c.RandString()},
|
|
}
|
|
},
|
|
func(pm map[string]docker.PortMapping, c fuzz.Continue) {
|
|
// This is necessary because keys with nil values get omitted.
|
|
// TODO: Is this a bug?
|
|
pm[c.RandString()] = docker.PortMapping{
|
|
c.RandString(): c.RandString(),
|
|
}
|
|
},
|
|
func(q *resource.Quantity, c fuzz.Continue) {
|
|
// Real Quantity fuzz testing is done elsewhere;
|
|
// this limited subset of functionality survives
|
|
// round-tripping to v1beta1/2.
|
|
q.Amount = &inf.Dec{}
|
|
q.Format = resource.DecimalExponent
|
|
//q.Amount.SetScale(inf.Scale(-c.Intn(12)))
|
|
q.Amount.SetUnscaled(c.Int63n(1000))
|
|
},
|
|
func(p *api.PullPolicy, c fuzz.Continue) {
|
|
policies := []api.PullPolicy{api.PullAlways, api.PullNever, api.PullIfNotPresent}
|
|
*p = policies[c.Rand.Intn(len(policies))]
|
|
},
|
|
)
|
|
return f
|
|
}
|
|
|
|
func fuzzInternalObject(t *testing.T, forVersion string, item runtime.Object, seed int64) runtime.Object {
|
|
fuzzerFor(t, forVersion, rand.NewSource(seed)).Fuzz(item)
|
|
|
|
j, err := meta.TypeAccessor(item)
|
|
if err != nil {
|
|
t.Fatalf("Unexpected error %v for %#v", err, item)
|
|
}
|
|
j.SetKind("")
|
|
j.SetAPIVersion("")
|
|
|
|
return item
|
|
}
|
|
|
|
func roundTrip(t *testing.T, codec runtime.Codec, item runtime.Object) {
|
|
name := reflect.TypeOf(item).Elem().Name()
|
|
data, err := codec.Encode(item)
|
|
if err != nil {
|
|
t.Errorf("%v: %v (%#v)", name, err, item)
|
|
return
|
|
}
|
|
|
|
obj2, err := codec.Decode(data)
|
|
if err != nil {
|
|
t.Errorf("0: %v: %v\nCodec: %v\nData: %s\nSource: %#v", name, err, codec, string(data), item)
|
|
return
|
|
}
|
|
if !api.Semantic.DeepEqual(item, obj2) {
|
|
t.Errorf("1: %v: diff: %v\nCodec: %v\nData: %s\nSource: %#v\nFinal: %#v", name, util.ObjectGoPrintDiff(item, obj2), codec, string(data), item, obj2)
|
|
return
|
|
}
|
|
|
|
obj3 := reflect.New(reflect.TypeOf(item).Elem()).Interface().(runtime.Object)
|
|
err = codec.DecodeInto(data, obj3)
|
|
if err != nil {
|
|
t.Errorf("2: %v: %v", name, err)
|
|
return
|
|
}
|
|
if !api.Semantic.DeepEqual(item, obj3) {
|
|
t.Errorf("3: %v: diff: %v\nCodec: %v", name, util.ObjectDiff(item, obj3), codec)
|
|
return
|
|
}
|
|
}
|
|
|
|
// roundTripSame verifies the same source object is tested in all API versions.
|
|
func roundTripSame(t *testing.T, item runtime.Object) {
|
|
seed := rand.Int63()
|
|
fuzzInternalObject(t, "", item, seed)
|
|
roundTrip(t, v1beta1.Codec, item)
|
|
roundTrip(t, v1beta2.Codec, item)
|
|
fuzzInternalObject(t, "v1beta3", item, seed)
|
|
roundTrip(t, v1beta3.Codec, item)
|
|
}
|
|
|
|
func roundTripAll(t *testing.T, item runtime.Object) {
|
|
seed := rand.Int63()
|
|
roundTrip(t, v1beta1.Codec, fuzzInternalObject(t, "v1beta1", item, seed))
|
|
roundTrip(t, v1beta2.Codec, fuzzInternalObject(t, "v1beta2", item, seed))
|
|
roundTrip(t, v1beta3.Codec, fuzzInternalObject(t, "v1beta3", item, seed))
|
|
}
|
|
|
|
// For debugging problems
|
|
func TestSpecificKind(t *testing.T) {
|
|
api.Scheme.Log(t)
|
|
defer api.Scheme.Log(nil)
|
|
|
|
kind := "PodList"
|
|
item, err := api.Scheme.New("", kind)
|
|
if err != nil {
|
|
t.Errorf("Couldn't make a %v? %v", kind, err)
|
|
return
|
|
}
|
|
roundTripSame(t, item)
|
|
}
|
|
|
|
func TestList(t *testing.T) {
|
|
api.Scheme.Log(t)
|
|
defer api.Scheme.Log(nil)
|
|
|
|
kind := "List"
|
|
item, err := api.Scheme.New("", kind)
|
|
if err != nil {
|
|
t.Errorf("Couldn't make a %v? %v", kind, err)
|
|
return
|
|
}
|
|
roundTripSame(t, item)
|
|
}
|
|
|
|
var nonRoundTrippableTypes = util.NewStringSet("ContainerManifest", "ContainerManifestList")
|
|
var nonInternalRoundTrippableTypes = util.NewStringSet("List")
|
|
|
|
func TestRoundTripTypes(t *testing.T) {
|
|
// api.Scheme.Log(t)
|
|
// defer api.Scheme.Log(nil)
|
|
|
|
for kind := range api.Scheme.KnownTypes("") {
|
|
if nonRoundTrippableTypes.Has(kind) {
|
|
continue
|
|
}
|
|
// Try a few times, since runTest uses random values.
|
|
for i := 0; i < *fuzzIters; i++ {
|
|
item, err := api.Scheme.New("", kind)
|
|
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)
|
|
}
|
|
roundTripSame(t, item)
|
|
if !nonInternalRoundTrippableTypes.Has(kind) {
|
|
roundTrip(t, api.Codec, fuzzInternalObject(t, "", item, rand.Int63()))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestEncode_Ptr(t *testing.T) {
|
|
pod := &api.Pod{
|
|
ObjectMeta: api.ObjectMeta{
|
|
Labels: map[string]string{"name": "foo"},
|
|
},
|
|
}
|
|
obj := runtime.Object(pod)
|
|
data, err := latest.Codec.Encode(obj)
|
|
obj2, err2 := latest.Codec.Decode(data)
|
|
if err != nil || err2 != nil {
|
|
t.Fatalf("Failure: '%v' '%v'", err, err2)
|
|
}
|
|
if _, ok := obj2.(*api.Pod); !ok {
|
|
t.Fatalf("Got wrong type")
|
|
}
|
|
if !api.Semantic.DeepEqual(obj2, pod) {
|
|
t.Errorf("Expected:\n %#v,\n Got:\n %#v", &pod, obj2)
|
|
}
|
|
}
|
|
|
|
func TestBadJSONRejection(t *testing.T) {
|
|
badJSONMissingKind := []byte(`{ }`)
|
|
if _, err := latest.Codec.Decode(badJSONMissingKind); err == nil {
|
|
t.Errorf("Did not reject despite lack of kind field: %s", badJSONMissingKind)
|
|
}
|
|
badJSONUnknownType := []byte(`{"kind": "bar"}`)
|
|
if _, err1 := latest.Codec.Decode(badJSONUnknownType); err1 == nil {
|
|
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)
|
|
}*/
|
|
}
|
|
|
|
const benchmarkSeed = 100
|
|
|
|
func BenchmarkEncode(b *testing.B) {
|
|
pod := api.Pod{}
|
|
apiObjectFuzzer := fuzzerFor(nil, "", rand.NewSource(benchmarkSeed))
|
|
apiObjectFuzzer.Fuzz(&pod)
|
|
for i := 0; i < b.N; i++ {
|
|
latest.Codec.Encode(&pod)
|
|
}
|
|
}
|
|
|
|
// BenchmarkEncodeJSON provides a baseline for regular JSON encode performance
|
|
func BenchmarkEncodeJSON(b *testing.B) {
|
|
pod := api.Pod{}
|
|
apiObjectFuzzer := fuzzerFor(nil, "", rand.NewSource(benchmarkSeed))
|
|
apiObjectFuzzer.Fuzz(&pod)
|
|
for i := 0; i < b.N; i++ {
|
|
json.Marshal(&pod)
|
|
}
|
|
}
|
|
|
|
func BenchmarkDecode(b *testing.B) {
|
|
pod := api.Pod{}
|
|
apiObjectFuzzer := fuzzerFor(nil, "", rand.NewSource(benchmarkSeed))
|
|
apiObjectFuzzer.Fuzz(&pod)
|
|
data, _ := latest.Codec.Encode(&pod)
|
|
for i := 0; i < b.N; i++ {
|
|
latest.Codec.Decode(data)
|
|
}
|
|
}
|
|
|
|
func BenchmarkDecodeInto(b *testing.B) {
|
|
pod := api.Pod{}
|
|
apiObjectFuzzer := fuzzerFor(nil, "", rand.NewSource(benchmarkSeed))
|
|
apiObjectFuzzer.Fuzz(&pod)
|
|
data, _ := latest.Codec.Encode(&pod)
|
|
for i := 0; i < b.N; i++ {
|
|
obj := api.Pod{}
|
|
latest.Codec.DecodeInto(data, &obj)
|
|
}
|
|
}
|
|
|
|
// BenchmarkDecodeJSON provides a baseline for regular JSON decode performance
|
|
func BenchmarkDecodeJSON(b *testing.B) {
|
|
pod := api.Pod{}
|
|
apiObjectFuzzer := fuzzerFor(nil, "", rand.NewSource(benchmarkSeed))
|
|
apiObjectFuzzer.Fuzz(&pod)
|
|
data, _ := latest.Codec.Encode(&pod)
|
|
for i := 0; i < b.N; i++ {
|
|
obj := api.Pod{}
|
|
json.Unmarshal(data, &obj)
|
|
}
|
|
}
|