kubernetes/pkg/util/time_cache_test.go
2015-01-13 18:09:13 -08:00

184 lines
4.5 KiB
Go

/*
Copyright 2015 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 util
import (
"math/rand"
"runtime"
"sync"
"testing"
"time"
fuzz "github.com/google/gofuzz"
)
func TestCacheExpire(t *testing.T) {
calls := map[string]int{}
ff := func(key string) T { calls[key]++; return key }
clock := &FakeClock{time.Now()}
c := NewTimeCache(clock, 60*time.Second, ff)
c.Get("foo")
c.Get("bar")
// This call should hit the cache, so we expect no additional calls
c.Get("foo")
// Advance the clock, this call should miss the cache, so expect one more call.
clock.Time = clock.Time.Add(61 * time.Second)
c.Get("foo")
c.Get("bar")
if e, a := 2, calls["foo"]; e != a {
t.Errorf("Wrong number of calls for foo: wanted %v, got %v", e, a)
}
if e, a := 2, calls["bar"]; e != a {
t.Errorf("Wrong number of calls for bar: wanted %v, got %v", e, a)
}
}
func TestCacheNotExpire(t *testing.T) {
calls := map[string]int{}
ff := func(key string) T { calls[key]++; return key }
clock := &FakeClock{time.Now()}
c := NewTimeCache(clock, 60*time.Second, ff)
c.Get("foo")
// This call should hit the cache, so we expect no additional calls to the cloud
clock.Time = clock.Time.Add(60 * time.Second)
c.Get("foo")
if e, a := 1, calls["foo"]; e != a {
t.Errorf("Wrong number of calls for foo: wanted %v, got %v", e, a)
}
}
func TestCacheParallel(t *testing.T) {
ff := func(key string) T { time.Sleep(time.Second); return key }
clock := &FakeClock{time.Now()}
c := NewTimeCache(clock, 60*time.Second, ff)
// Make some keys
keys := []string{}
fuzz.New().NilChance(0).NumElements(50, 50).Fuzz(&keys)
// If we have high parallelism, this will take only a second.
var wg sync.WaitGroup
wg.Add(len(keys))
for _, key := range keys {
go func(key string) {
c.Get(key)
wg.Done()
}(key)
}
wg.Wait()
}
func TestCacheParallelOneCall(t *testing.T) {
calls := 0
var callLock sync.Mutex
ff := func(key string) T {
time.Sleep(time.Second)
callLock.Lock()
defer callLock.Unlock()
calls++
return key
}
clock := &FakeClock{time.Now()}
c := NewTimeCache(clock, 60*time.Second, ff)
// If we have high parallelism, this will take only a second.
var wg sync.WaitGroup
wg.Add(50)
for i := 0; i < 50; i++ {
go func(key string) {
c.Get(key)
wg.Done()
}("aoeu")
}
wg.Wait()
// And if we wait for existing calls, we should have only one call.
if e, a := 1, calls; e != a {
t.Errorf("Expected %v, got %v", e, a)
}
}
func TestCacheParallelNoDeadlocksNoDoubleCalls(t *testing.T) {
// Make 50 random keys
keys := []string{}
fuzz.New().NilChance(0).NumElements(50, 50).Fuzz(&keys)
// Data structure for tracking when each key is accessed.
type callTrack struct {
sync.Mutex
accessTimes []time.Time
}
calls := map[string]*callTrack{}
for _, k := range keys {
calls[k] = &callTrack{}
}
// This is called to fill the cache in the case of a cache miss
// or cache entry expiration. We record the time.
ff := func(key string) T {
ct := calls[key]
ct.Lock()
ct.accessTimes = append(ct.accessTimes, time.Now())
ct.Unlock()
// make sure that there is time for multiple requests to come in
// for the same key before this returns.
time.Sleep(time.Millisecond)
return key
}
cacheDur := 10 * time.Millisecond
c := NewTimeCache(RealClock{}, cacheDur, ff)
// Spawn a bunch of goroutines, each of which sequentially requests
// 500 random keys from the cache.
runtime.GOMAXPROCS(16)
var wg sync.WaitGroup
for i := 0; i < 500; i++ {
wg.Add(1)
go func(seed int64) {
r := rand.New(rand.NewSource(seed))
for i := 0; i < 500; i++ {
c.Get(keys[r.Intn(len(keys))])
}
wg.Done()
}(rand.Int63())
}
wg.Wait()
// Since the cache should hold things for 10ms, no calls for a given key
// should be more closely spaced than that.
for k, ct := range calls {
if len(ct.accessTimes) < 2 {
continue
}
cur := ct.accessTimes[0]
for i := 1; i < len(ct.accessTimes); i++ {
next := ct.accessTimes[i]
if next.Sub(cur) < cacheDur {
t.Errorf("%v was called at %v and %v", k, cur, next)
}
cur = next
}
}
}