/* 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() }