package ccache import ( "strconv" "testing" "time" . "github.com/karlseguin/expect" ) type CacheTests struct{} func Test_Cache(t *testing.T) { Expectify(new(CacheTests), t) } func (_ CacheTests) DeletesAValue() { cache := New(Configure()) Expect(cache.ItemCount()).To.Equal(0) cache.Set("spice", "flow", time.Minute) cache.Set("worm", "sand", time.Minute) Expect(cache.ItemCount()).To.Equal(2) cache.Delete("spice") Expect(cache.Get("spice")).To.Equal(nil) Expect(cache.Get("worm").Value()).To.Equal("sand") Expect(cache.ItemCount()).To.Equal(1) } func (_ CacheTests) DeletesAPrefix() { cache := New(Configure()) Expect(cache.ItemCount()).To.Equal(0) cache.Set("aaa", "1", time.Minute) cache.Set("aab", "2", time.Minute) cache.Set("aac", "3", time.Minute) cache.Set("ac", "4", time.Minute) cache.Set("z5", "7", time.Minute) Expect(cache.ItemCount()).To.Equal(5) Expect(cache.DeletePrefix("9a")).To.Equal(0) Expect(cache.ItemCount()).To.Equal(5) Expect(cache.DeletePrefix("aa")).To.Equal(3) Expect(cache.Get("aaa")).To.Equal(nil) Expect(cache.Get("aab")).To.Equal(nil) Expect(cache.Get("aac")).To.Equal(nil) Expect(cache.Get("ac").Value()).To.Equal("4") Expect(cache.Get("z5").Value()).To.Equal("7") Expect(cache.ItemCount()).To.Equal(2) } func (_ CacheTests) OnDeleteCallbackCalled() { onDeleteFnCalled := false onDeleteFn := func(item *Item) { if item.key == "spice" { onDeleteFnCalled = true } } cache := New(Configure().OnDelete(onDeleteFn)) cache.Set("spice", "flow", time.Minute) cache.Set("worm", "sand", time.Minute) time.Sleep(time.Millisecond * 10) // Run once to init cache.Delete("spice") time.Sleep(time.Millisecond * 10) // Wait for worker to pick up deleted items Expect(cache.Get("spice")).To.Equal(nil) Expect(cache.Get("worm").Value()).To.Equal("sand") Expect(onDeleteFnCalled).To.Equal(true) } func (_ CacheTests) FetchesExpiredItems() { cache := New(Configure()) fn := func() (interface{}, error) { return "moo-moo", nil } cache.Set("beef", "moo", time.Second*-1) Expect(cache.Get("beef").Value()).To.Equal("moo") out, _ := cache.Fetch("beef", time.Second, fn) Expect(out.Value()).To.Equal("moo-moo") } func (_ CacheTests) GCsTheOldestItems() { cache := New(Configure().ItemsToPrune(10)) for i := 0; i < 500; i++ { cache.Set(strconv.Itoa(i), i, time.Minute) } //let the items get promoted (and added to our list) time.Sleep(time.Millisecond * 10) gcCache(cache) Expect(cache.Get("9")).To.Equal(nil) Expect(cache.Get("10").Value()).To.Equal(10) Expect(cache.ItemCount()).To.Equal(490) } func (_ CacheTests) PromotedItemsDontGetPruned() { cache := New(Configure().ItemsToPrune(10).GetsPerPromote(1)) for i := 0; i < 500; i++ { cache.Set(strconv.Itoa(i), i, time.Minute) } time.Sleep(time.Millisecond * 10) //run the worker once to init the list cache.Get("9") time.Sleep(time.Millisecond * 10) gcCache(cache) Expect(cache.Get("9").Value()).To.Equal(9) Expect(cache.Get("10")).To.Equal(nil) Expect(cache.Get("11").Value()).To.Equal(11) } func (_ CacheTests) TrackerDoesNotCleanupHeldInstance() { cache := New(Configure().ItemsToPrune(10).Track()) for i := 0; i < 10; i++ { cache.Set(strconv.Itoa(i), i, time.Minute) } item := cache.TrackingGet("0") time.Sleep(time.Millisecond * 10) gcCache(cache) Expect(cache.Get("0").Value()).To.Equal(0) Expect(cache.Get("1")).To.Equal(nil) item.Release() gcCache(cache) Expect(cache.Get("0")).To.Equal(nil) } func (_ CacheTests) RemovesOldestItemWhenFull() { onDeleteFnCalled := false onDeleteFn := func(item *Item) { if item.key == "0" { onDeleteFnCalled = true } } cache := New(Configure().MaxSize(5).ItemsToPrune(1).OnDelete(onDeleteFn)) for i := 0; i < 7; i++ { cache.Set(strconv.Itoa(i), i, time.Minute) } time.Sleep(time.Millisecond * 10) Expect(cache.Get("0")).To.Equal(nil) Expect(cache.Get("1")).To.Equal(nil) Expect(cache.Get("2").Value()).To.Equal(2) Expect(onDeleteFnCalled).To.Equal(true) Expect(cache.ItemCount()).To.Equal(5) } func (_ CacheTests) RemovesOldestItemWhenFullBySizer() { cache := New(Configure().MaxSize(9).ItemsToPrune(2)) for i := 0; i < 7; i++ { cache.Set(strconv.Itoa(i), &SizedItem{i, 2}, time.Minute) } time.Sleep(time.Millisecond * 10) Expect(cache.Get("0")).To.Equal(nil) Expect(cache.Get("1")).To.Equal(nil) Expect(cache.Get("2")).To.Equal(nil) Expect(cache.Get("3")).To.Equal(nil) Expect(cache.Get("4").Value().(*SizedItem).id).To.Equal(4) } func (_ CacheTests) SetUpdatesSizeOnDelta() { cache := New(Configure()) cache.Set("a", &SizedItem{0, 2}, time.Minute) cache.Set("b", &SizedItem{0, 3}, time.Minute) time.Sleep(time.Millisecond * 5) checkSize(cache, 5) cache.Set("b", &SizedItem{0, 3}, time.Minute) time.Sleep(time.Millisecond * 5) checkSize(cache, 5) cache.Set("b", &SizedItem{0, 4}, time.Minute) time.Sleep(time.Millisecond * 5) checkSize(cache, 6) cache.Set("b", &SizedItem{0, 2}, time.Minute) time.Sleep(time.Millisecond * 5) checkSize(cache, 4) cache.Delete("b") time.Sleep(time.Millisecond * 100) checkSize(cache, 2) } func (_ CacheTests) ReplaceDoesNotchangeSizeIfNotSet() { cache := New(Configure()) cache.Set("1", &SizedItem{1, 2}, time.Minute) cache.Set("2", &SizedItem{1, 2}, time.Minute) cache.Set("3", &SizedItem{1, 2}, time.Minute) cache.Replace("4", &SizedItem{1, 2}) time.Sleep(time.Millisecond * 5) checkSize(cache, 6) } func (_ CacheTests) ReplaceChangesSize() { cache := New(Configure()) cache.Set("1", &SizedItem{1, 2}, time.Minute) cache.Set("2", &SizedItem{1, 2}, time.Minute) cache.Replace("2", &SizedItem{1, 2}) time.Sleep(time.Millisecond * 5) checkSize(cache, 4) cache.Replace("2", &SizedItem{1, 1}) time.Sleep(time.Millisecond * 5) checkSize(cache, 3) cache.Replace("2", &SizedItem{1, 3}) time.Sleep(time.Millisecond * 5) checkSize(cache, 5) } type SizedItem struct { id int s int64 } func (s *SizedItem) Size() int64 { return s.s } func checkSize(cache *Cache, sz int64) { cache.Stop() Expect(cache.size).To.Equal(sz) cache.restart() } func gcCache(cache *Cache) { cache.Stop() cache.gc() cache.restart() }