apiserver/pkg/storage/cacher/caching_object_test.go

161 lines
4.3 KiB
Go

/*
Copyright 2019 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 cacher
import (
"bytes"
"fmt"
"io"
"sync"
"sync/atomic"
"testing"
v1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/api/meta"
"k8s.io/apimachinery/pkg/runtime"
)
type mockEncoder struct {
identifier runtime.Identifier
expectedResult string
expectedError error
callsNumber int32
}
func newMockEncoder(id, result string, err error) *mockEncoder {
return &mockEncoder{
identifier: runtime.Identifier(id),
expectedResult: result,
expectedError: err,
}
}
func (e *mockEncoder) encode(_ runtime.Object, w io.Writer) error {
atomic.AddInt32(&e.callsNumber, 1)
if e.expectedError != nil {
return e.expectedError
}
_, err := w.Write([]byte(e.expectedResult))
return err
}
func TestCachingObject(t *testing.T) {
object, err := newCachingObject(&v1.Pod{})
if err != nil {
t.Fatalf("couldn't create cachingObject: %v", err)
}
encoders := []*mockEncoder{
newMockEncoder("1", "result1", nil),
newMockEncoder("2", "", fmt.Errorf("mock error")),
newMockEncoder("3", "result3", nil),
}
for _, encoder := range encoders {
buffer := bytes.NewBuffer(nil)
err := object.CacheEncode(encoder.identifier, encoder.encode, buffer)
if a, e := err, encoder.expectedError; e != a {
t.Errorf("%s: unexpected error: %v, expected: %v", encoder.identifier, a, e)
}
if a, e := buffer.String(), encoder.expectedResult; e != a {
t.Errorf("%s: unexpected result: %s, expected: %s", encoder.identifier, a, e)
}
}
for _, encoder := range encoders {
if encoder.callsNumber != 1 {
t.Errorf("%s: unexpected number of encode() calls: %d", encoder.identifier, encoder.callsNumber)
}
}
}
func TestSelfLink(t *testing.T) {
object, err := newCachingObject(&v1.Pod{})
if err != nil {
t.Fatalf("couldn't create cachingObject: %v", err)
}
selfLink := "selfLink"
object.SetSelfLink(selfLink)
encodeSelfLink := func(obj runtime.Object, w io.Writer) error {
accessor, err := meta.Accessor(obj)
if err != nil {
t.Fatalf("failed to get accessor for %#v: %v", obj, err)
}
_, err = w.Write([]byte(accessor.GetSelfLink()))
return err
}
buffer := bytes.NewBuffer(nil)
if err := object.CacheEncode("", encodeSelfLink, buffer); err != nil {
t.Errorf("unexpected error: %v", err)
}
if a, e := buffer.String(), selfLink; a != e {
t.Errorf("unexpected serialization: %s, expected: %s", a, e)
}
// GetObject should also set selfLink.
buffer.Reset()
if err := encodeSelfLink(object.GetObject(), buffer); err != nil {
t.Errorf("unexpected error: %v", err)
}
if a, e := buffer.String(), selfLink; a != e {
t.Errorf("unexpected serialization: %s, expected: %s", a, e)
}
}
func TestCachingObjectRaces(t *testing.T) {
object, err := newCachingObject(&v1.Pod{})
if err != nil {
t.Fatalf("couldn't create cachingObject: %v", err)
}
encoders := []*mockEncoder{}
for i := 0; i < 10; i++ {
encoder := newMockEncoder(fmt.Sprintf("%d", i), "result", nil)
encoders = append(encoders, encoder)
}
numWorkers := 1000
wg := &sync.WaitGroup{}
wg.Add(numWorkers)
for i := 0; i < numWorkers; i++ {
go func() {
defer wg.Done()
object.SetSelfLink("selfLink")
buffer := bytes.NewBuffer(nil)
for _, encoder := range encoders {
buffer.Reset()
if err := object.CacheEncode(encoder.identifier, encoder.encode, buffer); err != nil {
t.Errorf("unexpected error: %v", err)
}
if callsNumber := atomic.LoadInt32(&encoder.callsNumber); callsNumber != 1 {
t.Errorf("unexpected number of serializations: %d", callsNumber)
}
}
accessor, err := meta.Accessor(object.GetObject())
if err != nil {
t.Fatalf("failed to get accessor: %v", err)
}
if selfLink := accessor.GetSelfLink(); selfLink != "selfLink" {
t.Errorf("unexpected selfLink: %s", selfLink)
}
}()
}
wg.Wait()
}