mirror of https://github.com/grpc/grpc-go.git
263 lines
7.0 KiB
Go
263 lines
7.0 KiB
Go
/*
|
|
*
|
|
* Copyright 2020 gRPC 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 cache
|
|
|
|
import (
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
"github.com/google/go-cmp/cmp/cmpopts"
|
|
)
|
|
|
|
const (
|
|
defaultTestCacheSize = 5
|
|
defaultTestCacheMaxSize = 1000000
|
|
defaultTestTimeout = 1 * time.Second
|
|
)
|
|
|
|
// TestGet verifies the Add and Get methods of cache.LRU.
|
|
func TestGet(t *testing.T) {
|
|
key1 := Key{Path: "/service1/method1", KeyMap: "k1=v1,k2=v2"}
|
|
key2 := Key{Path: "/service2/method2", KeyMap: "k1=v1,k2=v2"}
|
|
val1 := Entry{HeaderData: "h1=v1"}
|
|
val2 := Entry{HeaderData: "h2=v2"}
|
|
|
|
tests := []struct {
|
|
desc string
|
|
keysToAdd []Key
|
|
valsToAdd []*Entry
|
|
keyToGet Key
|
|
wantEntry *Entry
|
|
}{
|
|
{
|
|
desc: "Empty cache",
|
|
keyToGet: Key{},
|
|
},
|
|
{
|
|
desc: "Single entry miss",
|
|
keysToAdd: []Key{key1},
|
|
valsToAdd: []*Entry{&val1},
|
|
keyToGet: Key{},
|
|
},
|
|
{
|
|
desc: "Single entry hit",
|
|
keysToAdd: []Key{key1},
|
|
valsToAdd: []*Entry{&val1},
|
|
keyToGet: key1,
|
|
wantEntry: &val1,
|
|
},
|
|
{
|
|
desc: "Multi entry miss",
|
|
keysToAdd: []Key{key1, key2},
|
|
valsToAdd: []*Entry{&val1, &val2},
|
|
keyToGet: Key{},
|
|
},
|
|
{
|
|
desc: "Multi entry hit",
|
|
keysToAdd: []Key{key1, key2},
|
|
valsToAdd: []*Entry{&val1, &val2},
|
|
keyToGet: key1,
|
|
wantEntry: &val1,
|
|
},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
t.Run(test.desc, func(t *testing.T) {
|
|
lru := NewLRU(defaultTestCacheMaxSize, nil)
|
|
for i, key := range test.keysToAdd {
|
|
lru.Add(key, test.valsToAdd[i])
|
|
}
|
|
opts := []cmp.Option{
|
|
cmpopts.IgnoreInterfaces(struct{ sync.Locker }{}),
|
|
cmpopts.IgnoreUnexported(Entry{}),
|
|
}
|
|
if gotEntry := lru.Get(test.keyToGet); !cmp.Equal(gotEntry, test.wantEntry, opts...) {
|
|
t.Errorf("lru.Get(%+v) = %+v, want %+v", test.keyToGet, gotEntry, test.wantEntry)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
// TestRemove verifies the Add and Remove methods of cache.LRU.
|
|
func TestRemove(t *testing.T) {
|
|
keys := []Key{
|
|
{Path: "/service1/method1", KeyMap: "k1=v1,k2=v2"},
|
|
{Path: "/service2/method2", KeyMap: "k1=v1,k2=v2"},
|
|
{Path: "/service3/method3", KeyMap: "k1=v1,k2=v2"},
|
|
}
|
|
|
|
lru := NewLRU(defaultTestCacheMaxSize, nil)
|
|
for _, k := range keys {
|
|
lru.Add(k, &Entry{})
|
|
}
|
|
for _, k := range keys {
|
|
lru.Remove(k)
|
|
if entry := lru.Get(k); entry != nil {
|
|
t.Fatalf("lru.Get(%+v) after a call to lru.Remove succeeds, should have failed", k)
|
|
}
|
|
}
|
|
}
|
|
|
|
// TestExceedingSizeCausesEviction verifies the case where adding a new entry
|
|
// to the cache leads to eviction of old entries to make space for the new one.
|
|
func TestExceedingSizeCausesEviction(t *testing.T) {
|
|
evictCh := make(chan Key, defaultTestCacheSize)
|
|
onEvicted := func(k Key, _ *Entry) {
|
|
t.Logf("evicted key {%+v} from cache", k)
|
|
evictCh <- k
|
|
}
|
|
|
|
keysToFill := []Key{{Path: "a"}, {Path: "b"}, {Path: "c"}, {Path: "d"}, {Path: "e"}}
|
|
keysCausingEviction := []Key{{Path: "f"}, {Path: "g"}, {Path: "h"}, {Path: "i"}, {Path: "j"}}
|
|
|
|
lru := NewLRU(defaultTestCacheSize, onEvicted)
|
|
for _, key := range keysToFill {
|
|
lru.Add(key, &Entry{})
|
|
}
|
|
|
|
for i, key := range keysCausingEviction {
|
|
lru.Add(key, &Entry{})
|
|
|
|
timer := time.NewTimer(defaultTestTimeout)
|
|
select {
|
|
case <-timer.C:
|
|
t.Fatal("Test timeout waiting for eviction")
|
|
case k := <-evictCh:
|
|
timer.Stop()
|
|
if !cmp.Equal(k, keysToFill[i]) {
|
|
t.Fatalf("Evicted key %+v, wanted %+v", k, keysToFill[i])
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// TestAddCausesMultipleEvictions verifies the case where adding one new entry
|
|
// causes the eviction of multiple old entries to make space for the new one.
|
|
func TestAddCausesMultipleEvictions(t *testing.T) {
|
|
evictCh := make(chan Key, defaultTestCacheSize)
|
|
onEvicted := func(k Key, _ *Entry) {
|
|
evictCh <- k
|
|
}
|
|
|
|
keysToFill := []Key{{Path: "a"}, {Path: "b"}, {Path: "c"}, {Path: "d"}, {Path: "e"}}
|
|
keyCausingEviction := Key{Path: "abcde"}
|
|
|
|
lru := NewLRU(defaultTestCacheSize, onEvicted)
|
|
for _, key := range keysToFill {
|
|
lru.Add(key, &Entry{})
|
|
}
|
|
|
|
lru.Add(keyCausingEviction, &Entry{})
|
|
|
|
for i := range keysToFill {
|
|
timer := time.NewTimer(defaultTestTimeout)
|
|
select {
|
|
case <-timer.C:
|
|
t.Fatal("Test timeout waiting for eviction")
|
|
case k := <-evictCh:
|
|
timer.Stop()
|
|
if !cmp.Equal(k, keysToFill[i]) {
|
|
t.Fatalf("Evicted key %+v, wanted %+v", k, keysToFill[i])
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// TestModifyCausesMultipleEvictions verifies the case where mofiying an
|
|
// existing entry to increase its size leads to the eviction of older entries
|
|
// to make space for the new one.
|
|
func TestModifyCausesMultipleEvictions(t *testing.T) {
|
|
evictCh := make(chan Key, defaultTestCacheSize)
|
|
onEvicted := func(k Key, _ *Entry) {
|
|
evictCh <- k
|
|
}
|
|
|
|
keysToFill := []Key{{Path: "a"}, {Path: "b"}, {Path: "c"}, {Path: "d"}, {Path: "e"}}
|
|
lru := NewLRU(defaultTestCacheSize, onEvicted)
|
|
for _, key := range keysToFill {
|
|
lru.Add(key, &Entry{})
|
|
}
|
|
|
|
lru.Add(keysToFill[len(keysToFill)-1], &Entry{HeaderData: "xxxx"})
|
|
for i := range keysToFill[:len(keysToFill)-1] {
|
|
timer := time.NewTimer(defaultTestTimeout)
|
|
select {
|
|
case <-timer.C:
|
|
t.Fatal("Test timeout waiting for eviction")
|
|
case k := <-evictCh:
|
|
timer.Stop()
|
|
if !cmp.Equal(k, keysToFill[i]) {
|
|
t.Fatalf("Evicted key %+v, wanted %+v", k, keysToFill[i])
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestLRUResize(t *testing.T) {
|
|
tests := []struct {
|
|
desc string
|
|
maxSize int64
|
|
keysToFill []Key
|
|
newMaxSize int64
|
|
wantEvictedKeys []Key
|
|
}{
|
|
{
|
|
desc: "resize causes multiple evictions",
|
|
maxSize: 5,
|
|
keysToFill: []Key{{Path: "a"}, {Path: "b"}, {Path: "c"}, {Path: "d"}, {Path: "e"}},
|
|
newMaxSize: 3,
|
|
wantEvictedKeys: []Key{{Path: "a"}, {Path: "b"}},
|
|
},
|
|
{
|
|
desc: "resize causes no evictions",
|
|
maxSize: 50,
|
|
keysToFill: []Key{{Path: "a"}, {Path: "b"}, {Path: "c"}, {Path: "d"}, {Path: "e"}},
|
|
newMaxSize: 10,
|
|
wantEvictedKeys: []Key{},
|
|
},
|
|
{
|
|
desc: "resize to higher value",
|
|
maxSize: 5,
|
|
keysToFill: []Key{{Path: "a"}, {Path: "b"}, {Path: "c"}, {Path: "d"}, {Path: "e"}},
|
|
newMaxSize: 10,
|
|
wantEvictedKeys: []Key{},
|
|
},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
t.Run(test.desc, func(t *testing.T) {
|
|
var evictedKeys []Key
|
|
onEvicted := func(k Key, _ *Entry) {
|
|
evictedKeys = append(evictedKeys, k)
|
|
}
|
|
|
|
lru := NewLRU(test.maxSize, onEvicted)
|
|
for _, key := range test.keysToFill {
|
|
lru.Add(key, &Entry{})
|
|
}
|
|
lru.Resize(test.newMaxSize)
|
|
if !cmp.Equal(evictedKeys, test.wantEvictedKeys, cmpopts.EquateEmpty()) {
|
|
t.Fatalf("lru.Resize evicted keys {%v}, should have evicted {%v}", evictedKeys, test.wantEvictedKeys)
|
|
}
|
|
})
|
|
}
|
|
}
|