Skip to content

Commit bde4250

Browse files
authoredDec 20, 2022
Fix LRU to correctly update expire on value replace (#56)
1 parent fdad20a commit bde4250

File tree

5 files changed

+159
-28
lines changed

5 files changed

+159
-28
lines changed
 

‎go.mod

+13-4
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,19 @@
11
module github.com/mailgun/groupcache/v2
22

3+
go 1.19
4+
35
require (
4-
github.com/golang/protobuf v1.3.1
6+
github.com/golang/protobuf v1.5.2
57
github.com/segmentio/fasthash v1.0.3
6-
github.com/sirupsen/logrus v1.6.0
7-
golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a // indirect
8+
github.com/sirupsen/logrus v1.9.0
9+
github.com/stretchr/testify v1.8.1
810
)
911

10-
go 1.15
12+
require (
13+
github.com/davecgh/go-spew v1.1.1 // indirect
14+
github.com/pmezard/go-difflib v1.0.0 // indirect
15+
golang.org/x/sys v0.0.0-20220919091848-fb04ddd9f9c8 // indirect
16+
golang.org/x/xerrors v0.0.0-20220609144429-65e65417b02f // indirect
17+
google.golang.org/protobuf v1.28.1 // indirect
18+
gopkg.in/yaml.v3 v3.0.1 // indirect
19+
)

‎go.sum

+31-12
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,37 @@
1+
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
12
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
23
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
3-
github.com/golang/protobuf v1.3.1 h1:YF8+flBXS5eO826T4nzqPrxfhQThhXl0YzfuUPu4SBg=
4-
github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
5-
github.com/konsorten/go-windows-terminal-sequences v1.0.3 h1:CE8S1cTafDpPvMhIxNJKvHsGVBgn1xWYf1NbHQhywc8=
6-
github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
4+
github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
5+
github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw=
6+
github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY=
7+
github.com/google/go-cmp v0.5.5 h1:Khx7svrCpmxxtHBq5j2mp/xVjsi8hQMfNLvJFAlrGgU=
8+
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
79
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
810
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
911
github.com/segmentio/fasthash v1.0.3 h1:EI9+KE1EwvMLBWwjpRDc+fEM+prwxDYbslddQGtrmhM=
1012
github.com/segmentio/fasthash v1.0.3/go.mod h1:waKX8l2N8yckOgmSsXJi7x1ZfdKZ4x7KRMzBtS3oedY=
11-
github.com/sirupsen/logrus v1.6.0 h1:UBcNElsrwanuuMsnGSlYmtmgbb23qDR5dG+6X6Oo89I=
12-
github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88=
13-
github.com/stretchr/testify v1.2.2 h1:bSDNvY7ZPG5RlJ8otE/7V6gMiyenm9RtJ7IUVIAoJ1w=
14-
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
15-
golang.org/x/sys v0.0.0-20190422165155-953cdadca894 h1:Cz4ceDQGXuKRnVBDTS23GTn/pU5OE2C0WrNTOYK1Uuc=
16-
golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
17-
golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a h1:dGzPydgVsqGcTRVwiLJ1jVbufYwmzD3LfVPLKsKg+0k=
18-
golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
13+
github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0=
14+
github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
15+
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
16+
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
17+
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
18+
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
19+
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
20+
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
21+
github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk=
22+
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
23+
golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
24+
golang.org/x/sys v0.0.0-20220919091848-fb04ddd9f9c8 h1:h+EGohizhe9XlX18rfpa8k8RAc5XyaeamM+0VHRd4lc=
25+
golang.org/x/sys v0.0.0-20220919091848-fb04ddd9f9c8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
26+
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
27+
golang.org/x/xerrors v0.0.0-20220609144429-65e65417b02f h1:uF6paiQQebLeSXkrTqHqz0MXhXXS1KgF41eUdBNvxK0=
28+
golang.org/x/xerrors v0.0.0-20220609144429-65e65417b02f/go.mod h1:K8+ghG5WaK9qNqU5K3HdILfMLy1f3aNYFI/wnl100a8=
29+
google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
30+
google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
31+
google.golang.org/protobuf v1.28.1 h1:d0NfwRgPtno5B1Wa6L2DAG+KivqkdutMf1UhdNx175w=
32+
google.golang.org/protobuf v1.28.1/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
33+
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
34+
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
35+
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
36+
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
37+
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=

‎groupcache.go

+6
Original file line numberDiff line numberDiff line change
@@ -582,6 +582,11 @@ func (g *Group) CacheStats(which CacheType) CacheStats {
582582
}
583583
}
584584

585+
// NowFunc returns the current time which is used by the LRU to
586+
// determine if the value has expired. This can be overridden by
587+
// tests to ensure items are evicted when expired.
588+
var NowFunc lru.NowFunc = time.Now
589+
585590
// cache is a wrapper around an *lru.Cache that adds synchronization,
586591
// makes values always be ByteView, and counts the size of all keys and
587592
// values.
@@ -610,6 +615,7 @@ func (c *cache) add(key string, value ByteView) {
610615
defer c.mu.Unlock()
611616
if c.lru == nil {
612617
c.lru = &lru.Cache{
618+
Now: NowFunc,
613619
OnEvicted: func(key lru.Key, value interface{}) {
614620
val := value.(ByteView)
615621
c.nbytes -= int64(len(key.(string))) + int64(val.Len())

‎groupcache_test.go

+99-11
Original file line numberDiff line numberDiff line change
@@ -30,9 +30,9 @@ import (
3030
"unsafe"
3131

3232
"github.com/golang/protobuf/proto"
33-
3433
pb "github.com/mailgun/groupcache/v2/groupcachepb"
3534
"github.com/mailgun/groupcache/v2/testpb"
35+
"github.com/stretchr/testify/assert"
3636
)
3737

3838
var (
@@ -536,17 +536,105 @@ func TestContextDeadlineOnPeer(t *testing.T) {
536536
}
537537
}
538538

539-
func TestCacheRemovesBytesOfReplacedValues(t *testing.T) {
540-
c := cache{}
539+
func TestEnsureSizeReportedCorrectly(t *testing.T) {
540+
c := &cache{}
541541

542-
const key = "test"
543-
keyLen := len(key)
542+
// Add the first value
543+
bv1 := ByteView{s: "first", e: time.Now().Add(100 * time.Second)}
544+
c.add("key1", bv1)
545+
v, ok := c.get("key1")
544546

545-
for _, bytes := range []int{100, 1000, 2000} {
546-
c.add(key, ByteView{b: make([]byte, bytes)})
547-
expected := int64(bytes + keyLen)
548-
if c.nbytes != expected {
549-
t.Fatalf("%s: expected %d was %d", t.Name(), expected, c.nbytes)
550-
}
547+
// Should be len("key1" + "first") == 9
548+
assert.True(t, ok)
549+
assert.True(t, v.Equal(bv1))
550+
assert.Equal(t, int64(9), c.bytes())
551+
552+
// Add a second value
553+
bv2 := ByteView{s: "second", e: time.Now().Add(200 * time.Second)}
554+
555+
c.add("key2", bv2)
556+
v, ok = c.get("key2")
557+
558+
// Should be len("key2" + "second") == (10 + 9) == 19
559+
assert.True(t, ok)
560+
assert.True(t, v.Equal(bv2))
561+
assert.Equal(t, int64(19), c.bytes())
562+
563+
// Replace the first value with a shorter value
564+
bv3 := ByteView{s: "3", e: time.Now().Add(200 * time.Second)}
565+
566+
c.add("key1", bv3)
567+
v, ok = c.get("key1")
568+
569+
// len("key1" + "3") == 5
570+
// len("key2" + "second") == 10
571+
assert.True(t, ok)
572+
assert.True(t, v.Equal(bv3))
573+
assert.Equal(t, int64(15), c.bytes())
574+
575+
// Replace the second value with a longer value
576+
bv4 := ByteView{s: "this-string-is-28-chars-long", e: time.Now().Add(200 * time.Second)}
577+
578+
c.add("key2", bv4)
579+
v, ok = c.get("key2")
580+
581+
// len("key1" + "3") == 5
582+
// len("key2" + "this-string-is-28-chars-long") == 32
583+
assert.True(t, ok)
584+
assert.True(t, v.Equal(bv4))
585+
assert.Equal(t, int64(37), c.bytes())
586+
}
587+
588+
func TestEnsureUpdateExpiredValue(t *testing.T) {
589+
c := &cache{}
590+
curTime := time.Now()
591+
592+
// Override the now function so we control time
593+
NowFunc = func() time.Time {
594+
return curTime
551595
}
596+
597+
// Expires in 1 second
598+
c.add("key1", ByteView{s: "value1", e: curTime.Add(time.Second)})
599+
_, ok := c.get("key1")
600+
assert.True(t, ok)
601+
602+
// Advance 1.1 seconds into the future
603+
curTime = curTime.Add(time.Millisecond * 1100)
604+
605+
// Value should have expired
606+
_, ok = c.get("key1")
607+
assert.False(t, ok)
608+
609+
// Add a new key that expires in 1 second
610+
c.add("key2", ByteView{s: "value2", e: curTime.Add(time.Second)})
611+
_, ok = c.get("key2")
612+
assert.True(t, ok)
613+
614+
// Advance 0.5 seconds into the future
615+
curTime = curTime.Add(time.Millisecond * 500)
616+
617+
// Value should still exist
618+
_, ok = c.get("key2")
619+
assert.True(t, ok)
620+
621+
// Replace the existing key, this should update the expired time
622+
c.add("key2", ByteView{s: "updated value2", e: curTime.Add(time.Second)})
623+
_, ok = c.get("key2")
624+
assert.True(t, ok)
625+
626+
// Advance 0.6 seconds into the future, which puts us past the initial
627+
// expired time for key2.
628+
curTime = curTime.Add(time.Millisecond * 600)
629+
630+
// Should still exist
631+
_, ok = c.get("key2")
632+
assert.True(t, ok)
633+
634+
// Advance 1.1 seconds into the future
635+
curTime = curTime.Add(time.Millisecond * 1100)
636+
637+
// Should not exist
638+
_, ok = c.get("key2")
639+
assert.False(t, ok)
552640
}

‎lru/lru.go

+10-1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,8 @@ import (
2222
"time"
2323
)
2424

25+
type NowFunc func() time.Time
26+
2527
// Cache is an LRU cache. It is not safe for concurrent access.
2628
type Cache struct {
2729
// MaxEntries is the maximum number of cache entries before
@@ -32,6 +34,11 @@ type Cache struct {
3234
// executed when an entry is purged from the cache.
3335
OnEvicted func(key Key, value interface{})
3436

37+
// Now is the Now() function the cache will use to determine
38+
// the current time which is used to calculate expired values
39+
// Defaults to time.Now()
40+
Now NowFunc
41+
3542
ll *list.List
3643
cache map[interface{}]*list.Element
3744
}
@@ -53,6 +60,7 @@ func New(maxEntries int) *Cache {
5360
MaxEntries: maxEntries,
5461
ll: list.New(),
5562
cache: make(map[interface{}]*list.Element),
63+
Now: time.Now,
5664
}
5765
}
5866

@@ -68,6 +76,7 @@ func (c *Cache) Add(key Key, value interface{}, expire time.Time) {
6876
c.OnEvicted(key, eee.value)
6977
}
7078
c.ll.MoveToFront(ee)
79+
eee.expire = expire
7180
eee.value = value
7281
return
7382
}
@@ -86,7 +95,7 @@ func (c *Cache) Get(key Key) (value interface{}, ok bool) {
8695
if ele, hit := c.cache[key]; hit {
8796
entry := ele.Value.(*entry)
8897
// If the entry has expired, remove it from the cache
89-
if !entry.expire.IsZero() && entry.expire.Before(time.Now()) {
98+
if !entry.expire.IsZero() && entry.expire.Before(c.Now()) {
9099
c.removeElement(ele)
91100
return nil, false
92101
}

0 commit comments

Comments
 (0)
Please sign in to comment.