From a2453a4a1f3387a130e0c1291ebbdd7c13ba7448 Mon Sep 17 00:00:00 2001 From: FishGoddess <1149062639@qq.com> Date: Fri, 12 Jan 2024 16:40:35 +0800 Subject: [PATCH 1/6] =?UTF-8?q?=E7=B2=BE=E7=AE=80=20loader=20=E8=AE=BE?= =?UTF-8?q?=E8=AE=A1?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .github/workflows/test.yml | 2 +- FUTURE.md | 1 + Makefile | 4 +- _icons/jetbrains.png | Bin 124736 -> 0 bytes cache.go | 91 +++++++++---------------------------- cache_test.go | 35 +++----------- cache_type.go | 52 +++++++++++++++++++++ cache_type_test.go | 44 ++++++++++++++++++ doc.go | 2 +- go.mod | 2 +- lfu.go | 24 ++++++++-- lfu_test.go | 4 ++ load.go | 57 +++++++---------------- load_test.go | 49 ++++++++++---------- lru.go | 24 ++++++++-- report_test.go | 2 + sharding.go | 9 ++-- sharding_test.go | 2 + standard.go | 23 ++++++++-- standard_test.go | 1 + 20 files changed, 249 insertions(+), 179 deletions(-) delete mode 100644 _icons/jetbrains.png create mode 100644 cache_type.go create mode 100644 cache_type_test.go diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index d790f05..2d53fdf 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -12,7 +12,7 @@ jobs: - name: Setup uses: actions/setup-go@v4 with: - go-version: "1.17" + go-version: "1.20" - run: go version - name: Checkout diff --git a/FUTURE.md b/FUTURE.md index 3ea524d..28eaeff 100644 --- a/FUTURE.md +++ b/FUTURE.md @@ -5,6 +5,7 @@ * [ ] ~~提供一个清空并设置全量值的方法,方便定时数据的全量替换~~ 目前还找不到一个合适的设计去加入这个功能,并且也不是非常刚需,通过业务手段可以处理,所以先不加 * [ ] 完善监控上报器,提供更多缓存信息查询的方法 +* [ ] 梳理代码,优化代码风格 ### v0.4.x diff --git a/Makefile b/Makefile index aebba3b..77225c3 100644 --- a/Makefile +++ b/Makefile @@ -3,10 +3,10 @@ all: test bench test: - go test -cover ./... + go test -cover -count=1 -test.cpu=1 ./... bench: - go test -v -bench=. -benchtime=1s ./_examples/performance_test.go + go test -v ./_examples/performance_test.go -bench=. -benchtime=1s fmt: go fmt ./... \ No newline at end of file diff --git a/_icons/jetbrains.png b/_icons/jetbrains.png deleted file mode 100644 index 713d21837825eba4ccb839d6e685d6053d07722c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 124736 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
IiMgUJk^d#bDdIAq#T2R>K;v+wkH~yA$<8 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 G-uYiJ%OV;8 diff --git a/cache.go b/cache.go index 4713cc7..e838fa4 100644 --- a/cache.go +++ b/cache.go @@ -16,7 +16,6 @@ package cachego import ( "context" - "sync" "time" "github.com/FishGoddess/cachego/pkg/task" @@ -27,20 +26,6 @@ const ( NoTTL = 0 ) -const ( - // standard cache is a simple cache with locked map. - // It evicts entries randomly if cache size reaches to max entries. - standard CacheType = "standard" - - // lru cache is a cache using lru to evict entries. - // More details see https://en.wikipedia.org/wiki/Cache_replacement_policies#Least_recently_used_(LRU). - lru CacheType = "lru" - - // lfu cache is a cache using lfu to evict entries. - // More details see https://en.wikipedia.org/wiki/Cache_replacement_policies#Least-frequently_used_(LFU). - lfu CacheType = "lfu" -) - var ( newCaches = map[CacheType]func(conf *config) Cache{ standard: newStandardCache, @@ -49,29 +34,6 @@ var ( } ) -// CacheType is the type of cache. -type CacheType string - -// String returns the cache type in string form. -func (ct CacheType) String() string { - return string(ct) -} - -// IsStandard returns if cache type is standard. -func (ct CacheType) IsStandard() bool { - return ct == standard -} - -// IsLRU returns if cache type is lru. -func (ct CacheType) IsLRU() bool { - return ct == lru -} - -// IsLFU returns if cache type is lfu. -func (ct CacheType) IsLFU() bool { - return ct == lfu -} - // Cache is the core interface of cachego. // We provide some implements including standard cache and sharding cache. type Cache interface { @@ -101,38 +63,10 @@ type Cache interface { // Reset resets cache to initial status which is like a new cache. Reset() - // Loader loads a value to cache. - // See Loader interface. - Loader -} - -type cache struct { - *config - Loader - - lock sync.RWMutex -} - -func (c *cache) setup(conf *config, cache Cache) { - c.config = conf - c.Loader = NewLoader(cache, conf.singleflight) -} - -// RunGCTask runs a gc task in a new goroutine and returns a cancel function to cancel the task. -// However, you don't need to call it manually for most time, instead, use options is a better choice. -// Making it a public function is for more customizations in some situations. -// For example, using options to run gc task is un-cancelable, so you can use it to run gc task by your own -// and get a cancel function to cancel the gc task. -func RunGCTask(cache Cache, duration time.Duration) (cancel func()) { - fn := func(ctx context.Context) { - cache.GC() - } - - ctx := context.Background() - ctx, cancel = context.WithCancel(ctx) - - go task.New(fn).Context(ctx).Duration(duration).Run() - return cancel + // Load loads a key with ttl to cache and returns an error if failed. + // We recommend you use this method to load missed keys to cache, + // because it may use singleflight to reduce the times calling load function. + Load(key string, ttl time.Duration, load func() (value interface{}, err error)) (value interface{}, err error) } func newCache(withReport bool, opts ...Option) (cache Cache, reporter *Reporter) { @@ -180,3 +114,20 @@ func NewCache(opts ...Option) (cache Cache) { func NewCacheWithReport(opts ...Option) (cache Cache, reporter *Reporter) { return newCache(true, opts...) } + +// RunGCTask runs a gc task in a new goroutine and returns a cancel function to cancel the task. +// However, you don't need to call it manually for most time, instead, use options is a better choice. +// Making it a public function is for more customizations in some situations. +// For example, using options to run gc task is un-cancelable, so you can use it to run gc task by your own +// and get a cancel function to cancel the gc task. +func RunGCTask(cache Cache, duration time.Duration) (cancel func()) { + fn := func(ctx context.Context) { + cache.GC() + } + + ctx := context.Background() + ctx, cancel = context.WithCancel(ctx) + + go task.New(fn).Context(ctx).Duration(duration).Run() + return cancel +} diff --git a/cache_test.go b/cache_test.go index b061db3..4172fbb 100644 --- a/cache_test.go +++ b/cache_test.go @@ -25,35 +25,10 @@ const ( maxTestEntries = 10 ) -// go test -v -cover -run=^TestCacheType$ -func TestCacheType(t *testing.T) { - if standard.String() != string(standard) { - t.Fatalf("standard.String() %s is wrong", standard.String()) - } - - if lru.String() != string(lru) { - t.Fatalf("lru.String() %s is wrong", lru.String()) - } - - if lfu.String() != string(lfu) { - t.Fatalf("lfu.String() %s is wrong", lfu.String()) - } - - if !standard.IsStandard() { - t.Fatal("!standard.IsStandard()") - } - - if !lru.IsLRU() { - t.Fatal("!standard.IsLRU()") - } - - if !lfu.IsLFU() { - t.Fatal("!standard.IsLFU()") - } -} - type testCache struct { - cache + *config + loader *Loader + count int32 } @@ -84,6 +59,10 @@ func (tc *testCache) GC() (cleans int) { func (tc *testCache) Reset() {} +func (tc *testCache) Load(key string, ttl time.Duration, load func() (value interface{}, err error)) (value interface{}, err error) { + return nil, nil +} + func testCacheGet(t *testing.T, cache Cache) { value, found := cache.Get("key") if found { diff --git a/cache_type.go b/cache_type.go new file mode 100644 index 0000000..95a65ea --- /dev/null +++ b/cache_type.go @@ -0,0 +1,52 @@ +// Copyright 2024 FishGoddess. 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 cachego + +const ( + // standard cache is a simple cache with locked map. + // It evicts entries randomly if cache size reaches to max entries. + standard CacheType = "standard" + + // lru cache is a cache using lru to evict entries. + // More details see https://en.wikipedia.org/wiki/Cache_replacement_policies#Least_recently_used_(LRU). + lru CacheType = "lru" + + // lfu cache is a cache using lfu to evict entries. + // More details see https://en.wikipedia.org/wiki/Cache_replacement_policies#Least-frequently_used_(LFU). + lfu CacheType = "lfu" +) + +// CacheType is the type of cache. +type CacheType string + +// String returns the cache type in string form. +func (ct CacheType) String() string { + return string(ct) +} + +// IsStandard returns if cache type is standard. +func (ct CacheType) IsStandard() bool { + return ct == standard +} + +// IsLRU returns if cache type is lru. +func (ct CacheType) IsLRU() bool { + return ct == lru +} + +// IsLFU returns if cache type is lfu. +func (ct CacheType) IsLFU() bool { + return ct == lfu +} diff --git a/cache_type_test.go b/cache_type_test.go new file mode 100644 index 0000000..fe82ce1 --- /dev/null +++ b/cache_type_test.go @@ -0,0 +1,44 @@ +// Copyright 2024 FishGoddess. 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 cachego + +import "testing" + +// go test -v -cover -run=^TestCacheType$ +func TestCacheType(t *testing.T) { + if standard.String() != string(standard) { + t.Fatalf("standard.String() %s is wrong", standard.String()) + } + + if lru.String() != string(lru) { + t.Fatalf("lru.String() %s is wrong", lru.String()) + } + + if lfu.String() != string(lfu) { + t.Fatalf("lfu.String() %s is wrong", lfu.String()) + } + + if !standard.IsStandard() { + t.Fatal("!standard.IsStandard()") + } + + if !lru.IsLRU() { + t.Fatal("!standard.IsLRU()") + } + + if !lfu.IsLFU() { + t.Fatal("!standard.IsLFU()") + } +} diff --git a/doc.go b/doc.go index 2fa3c4e..819a092 100644 --- a/doc.go +++ b/doc.go @@ -480,4 +480,4 @@ Package cachego provides an easy way to use foundation for your caching operatio package cachego // import "github.com/FishGoddess/cachego" // Version is the version string representation of cachego. -const Version = "v0.5.0" +const Version = "v0.6.0-alpha" diff --git a/go.mod b/go.mod index ee81265..a1f8805 100644 --- a/go.mod +++ b/go.mod @@ -1,3 +1,3 @@ module github.com/FishGoddess/cachego -go 1.17 +go 1.20 diff --git a/lfu.go b/lfu.go index 9727c7c..7a04e57 100644 --- a/lfu.go +++ b/lfu.go @@ -15,16 +15,20 @@ package cachego import ( + "sync" "time" "github.com/FishGoddess/cachego/pkg/heap" ) type lfuCache struct { - cache + *config itemMap map[string]*heap.Item itemHeap *heap.Heap + lock sync.RWMutex + + loader *Loader } func newLFUCache(conf *config) Cache { @@ -33,11 +37,12 @@ func newLFUCache(conf *config) Cache { } cache := &lfuCache{ + config: conf, itemMap: make(map[string]*heap.Item, mapInitialCap), itemHeap: heap.New(sliceInitialCap), + loader: NewLoader(conf.singleflight), } - cache.setup(conf, cache) return cache } @@ -137,7 +142,8 @@ func (lc *lfuCache) gc() (cleans int) { func (lc *lfuCache) reset() { lc.itemMap = make(map[string]*heap.Item, mapInitialCap) lc.itemHeap = heap.New(sliceInitialCap) - lc.Loader.Reset() + + lc.loader.Reset() } // Get gets the value of key from cache and returns value if found. @@ -193,3 +199,15 @@ func (lc *lfuCache) Reset() { lc.reset() } + +// Load loads a value by load function and sets it to cache. +// Returns an error if load failed. +func (lc *lfuCache) Load(key string, ttl time.Duration, load func() (value interface{}, err error)) (value interface{}, err error) { + value, err = lc.loader.Load(key, ttl, load) + if err != nil { + return value, err + } + + lc.Set(key, value, ttl) + return value, nil +} diff --git a/lfu_test.go b/lfu_test.go index 51a0de4..b4359e5 100644 --- a/lfu_test.go +++ b/lfu_test.go @@ -25,6 +25,7 @@ import ( func newTestLFUCache() *lfuCache { conf := newDefaultConfig() conf.maxEntries = maxTestEntries + return newLFUCache(conf).(*lfuCache) } @@ -54,6 +55,7 @@ func TestLFUCacheEvict(t *testing.T) { for i := cache.maxEntries*10 - cache.maxEntries; i < cache.maxEntries*10; i++ { for j := 0; j < i; j++ { data := strconv.Itoa(i) + cache.Set(data, data, time.Duration(i)*time.Second) cache.Get(data) } @@ -61,6 +63,7 @@ func TestLFUCacheEvict(t *testing.T) { for i := cache.maxEntries*10 - cache.maxEntries; i < cache.maxEntries*10; i++ { data := strconv.Itoa(i) + value, ok := cache.Get(data) if !ok || value.(string) != data { t.Fatalf("!ok %+v || value.(string) %s != data %s", !ok, value.(string), data) @@ -68,6 +71,7 @@ func TestLFUCacheEvict(t *testing.T) { } i := cache.maxEntries*10 - cache.maxEntries + for cache.itemHeap.Size() > 0 { item := cache.itemHeap.Pop() entry := item.Value.(*entry) diff --git a/load.go b/load.go index 054bfa3..5af0370 100644 --- a/load.go +++ b/load.go @@ -18,66 +18,41 @@ import ( "errors" "time" - "github.com/FishGoddess/cachego/pkg/singleflight" + flight "github.com/FishGoddess/cachego/pkg/singleflight" ) -// Loader loads a value to cache. -// All implements should store a cache inside in order to load value to cache. -type Loader interface { - // Load loads a key with ttl to cache and returns an error if failed. - // We recommend you use this method to load missed keys to cache because it may use singleflight to reduce the times calling load function. - Load(key string, ttl time.Duration, load func() (value interface{}, err error)) (value interface{}, err error) - - // Reset resets loader to initial status which is like a new loader. - Reset() +// Loader loads values from somewhere. +type Loader struct { + group *flight.Group } -// Loader loads a value to cache. -// We recommend you set enableSingleflight=true in NewLoader for reducing the times calling load function. -type loader struct { - cache Cache - group *singleflight.Group -} +// NewLoader creates a loader. +// It also creates a singleflight group to call load if singleflight is true. +func NewLoader(singleflight bool) *Loader { + loader := new(Loader) -// NewLoader creates a loader with cache. -// It also creates a singleflight group to call load if enableSingleflight is true. -func NewLoader(cache Cache, enableSingleflight bool) Loader { - loader := &loader{ - cache: cache, - } - - if enableSingleflight { - loader.group = singleflight.NewGroup(mapInitialCap) + if singleflight { + loader.group = flight.NewGroup(mapInitialCap) } return loader } -// Load loads a key with ttl to cache and returns an error if failed. -func (l *loader) Load(key string, ttl time.Duration, load func() (value interface{}, err error)) (value interface{}, err error) { +// Load loads a value of key with ttl and returns an error if failed. +func (l *Loader) Load(key string, ttl time.Duration, load func() (value interface{}, err error)) (value interface{}, err error) { if load == nil { return nil, errors.New("cachego: load function is nil in loader") } - if l.group != nil { - value, err = l.group.Call(key, load) - } else { - value, err = load() - } - - if err != nil { - return value, err - } - - if l.cache != nil { - l.cache.Set(key, value, ttl) + if l.group == nil { + return load() } - return value, err + return l.group.Call(key, load) } // Reset resets loader to initial status which is like a new loader. -func (l *loader) Reset() { +func (l *Loader) Reset() { if l.group != nil { l.group.Reset() } diff --git a/load_test.go b/load_test.go index 6275f01..b0948dd 100644 --- a/load_test.go +++ b/load_test.go @@ -26,13 +26,14 @@ type testLoadCache struct { value interface{} ttl time.Duration - loader Loader + loader *Loader } func newTestLoadCache(singleflight bool) Cache { - cache := new(testLoadCache) - loader := NewLoader(cache, singleflight) - cache.loader = loader + cache := &testLoadCache{ + loader: NewLoader(singleflight), + } + return cache } @@ -44,6 +45,7 @@ func (tlc *testLoadCache) Set(key string, value interface{}, ttl time.Duration) tlc.key = key tlc.value = value tlc.ttl = ttl + return nil } @@ -67,26 +69,14 @@ func (tlc *testLoadCache) Load(key string, ttl time.Duration, load func() (value // go test -v -cover -run=^TestNewLoader$ func TestNewLoader(t *testing.T) { - l := NewLoader(nil, false) - - loader1, ok := l.(*loader) - if !ok { - t.Fatalf("l.(*loader) %T not ok", l) - } - - if loader1.group != nil { - t.Fatalf("loader1.group %+v != nil", loader1.group) - } - - l = NewLoader(nil, true) - - loader2, ok := l.(*loader) - if !ok { - t.Fatalf("l.(*loader) %T not ok", l) + loader := NewLoader(false) + if loader.group != nil { + t.Fatalf("loader.group %+v != nil", loader.group) } - if loader2.group == nil { - t.Fatal("loader2.group == nil") + loader = NewLoader(true) + if loader.group == nil { + t.Fatal("loader.group == nil") } } @@ -119,7 +109,10 @@ func TestLoaderLoad(t *testing.T) { cache = newTestLoadCache(true) loadCount = 0 + var errs []error + var lock sync.Mutex var wg sync.WaitGroup + for i := int64(0); i < 100; i++ { wg.Add(1) @@ -131,16 +124,26 @@ func TestLoaderLoad(t *testing.T) { _, err := cache.Load("key", time.Duration(i), func() (value interface{}, err error) { time.Sleep(time.Second) loadCount++ + return str, nil }) if err != nil { - t.Fatal(err) + lock.Lock() + errs = append(errs, err) + lock.Unlock() } }(i) } wg.Wait() + + for _, err := range errs { + if err != nil { + t.Fatal(err) + } + } + if loadCount != 1 { t.Fatalf("loadCount %d != 1", loadCount) } diff --git a/lru.go b/lru.go index f2468ec..40bef21 100644 --- a/lru.go +++ b/lru.go @@ -16,14 +16,18 @@ package cachego import ( "container/list" + "sync" "time" ) type lruCache struct { - cache + *config elementMap map[string]*list.Element elementList *list.List + lock sync.RWMutex + + loader *Loader } func newLRUCache(conf *config) Cache { @@ -32,11 +36,12 @@ func newLRUCache(conf *config) Cache { } cache := &lruCache{ + config: conf, elementMap: make(map[string]*list.Element, mapInitialCap), elementList: list.New(), + loader: NewLoader(conf.singleflight), } - cache.setup(conf, cache) return cache } @@ -136,7 +141,8 @@ func (lc *lruCache) gc() (cleans int) { func (lc *lruCache) reset() { lc.elementMap = make(map[string]*list.Element, mapInitialCap) lc.elementList = list.New() - lc.Loader.Reset() + + lc.loader.Reset() } // Get gets the value of key from cache and returns value if found. @@ -192,3 +198,15 @@ func (lc *lruCache) Reset() { lc.reset() } + +// Load loads a value by load function and sets it to cache. +// Returns an error if load failed. +func (lc *lruCache) Load(key string, ttl time.Duration, load func() (value interface{}, err error)) (value interface{}, err error) { + value, err = lc.loader.Load(key, ttl, load) + if err != nil { + return value, err + } + + lc.Set(key, value, ttl) + return value, nil +} diff --git a/report_test.go b/report_test.go index 7f73ba7..607a8cf 100644 --- a/report_test.go +++ b/report_test.go @@ -130,6 +130,7 @@ func TestReportableCacheReportGC(t *testing.T) { gcCount := uint64(0) checked := false + cache.reportGC = func(reporter *Reporter, cost time.Duration, cleans int) { if cost <= 0 { t.Fatalf("cost %d <= 0", cost) @@ -165,6 +166,7 @@ func TestReportableCacheReportLoad(t *testing.T) { loadCount := uint64(0) checked := false + cache.reportLoad = func(reporter *Reporter, key string, value interface{}, ttl time.Duration, err error) { if key != "load" { t.Fatalf("key %s is wrong", key) diff --git a/sharding.go b/sharding.go index 77f7fdd..096e5c3 100644 --- a/sharding.go +++ b/sharding.go @@ -38,15 +38,18 @@ func newShardingCache(conf *config, newCache func(conf *config) Cache) Cache { caches = append(caches, newCache(conf)) } - return &shardingCache{ + cache := &shardingCache{ config: conf, caches: caches, } + + return cache } func (sc *shardingCache) cacheOf(key string) Cache { hash := sc.hash(key) mask := len(sc.caches) - 1 + return sc.caches[hash&mask] } @@ -95,8 +98,8 @@ func (sc *shardingCache) Reset() { } } -// Load loads a key with ttl to cache and returns an error if failed. -// See Cache interface. +// Load loads a value by load function and sets it to cache. +// Returns an error if load failed. func (sc *shardingCache) Load(key string, ttl time.Duration, load func() (value interface{}, err error)) (value interface{}, err error) { return sc.cacheOf(key).Load(key, ttl, load) } diff --git a/sharding_test.go b/sharding_test.go index ad26ac0..86f3f1f 100644 --- a/sharding_test.go +++ b/sharding_test.go @@ -26,6 +26,7 @@ const ( func newTestShardingCache() *shardingCache { conf := newDefaultConfig() conf.shardings = testShardings + return newShardingCache(conf, newStandardCache).(*shardingCache) } @@ -38,6 +39,7 @@ func TestShardingCache(t *testing.T) { // go test -v -cover -run=^TestShardingCacheIndex$ func TestShardingCacheIndex(t *testing.T) { cache := newTestShardingCache() + if len(cache.caches) != testShardings { t.Fatalf("len(cache.caches) %d is wrong", len(cache.caches)) } diff --git a/standard.go b/standard.go index 080cdd0..b2fa599 100644 --- a/standard.go +++ b/standard.go @@ -15,21 +15,26 @@ package cachego import ( + "sync" "time" ) type standardCache struct { - cache + *config entries map[string]*entry + lock sync.RWMutex + + loader *Loader } func newStandardCache(conf *config) Cache { cache := &standardCache{ + config: conf, entries: make(map[string]*entry, mapInitialCap), + loader: NewLoader(conf.singleflight), } - cache.setup(conf, cache) return cache } @@ -101,7 +106,7 @@ func (sc *standardCache) gc() (cleans int) { func (sc *standardCache) reset() { sc.entries = make(map[string]*entry, mapInitialCap) - sc.Loader.Reset() + sc.loader.Reset() } // Get gets the value of key from cache and returns value if found. @@ -157,3 +162,15 @@ func (sc *standardCache) Reset() { sc.reset() } + +// Load loads a value by load function and sets it to cache. +// Returns an error if load failed. +func (sc *standardCache) Load(key string, ttl time.Duration, load func() (value interface{}, err error)) (value interface{}, err error) { + value, err = sc.loader.Load(key, ttl, load) + if err != nil { + return value, err + } + + sc.Set(key, value, ttl) + return value, nil +} diff --git a/standard_test.go b/standard_test.go index d9bbfac..5bb82c2 100644 --- a/standard_test.go +++ b/standard_test.go @@ -23,6 +23,7 @@ import ( func newTestStandardCache() *standardCache { conf := newDefaultConfig() conf.maxEntries = maxTestEntries + return newStandardCache(conf).(*standardCache) } From e69b9412ca57aa97e697583c532dbb2c96e05771 Mon Sep 17 00:00:00 2001 From: FishGoddess <1149062639@qq.com> Date: Sat, 13 Jan 2024 20:42:15 +0800 Subject: [PATCH 2/6] =?UTF-8?q?=E8=B0=83=E6=95=B4=E5=8D=95=E6=B5=8B?= =?UTF-8?q?=E6=89=A7=E8=A1=8C=E5=91=BD=E4=BB=A4?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- FUTURE.md | 7 ++++-- HISTORY.md | 7 ++++++ README.en.md | 2 +- README.md | 2 +- _icons/coverage.svg | 4 +-- cache_test.go | 6 ++--- cache_type_test.go | 2 +- config_test.go | 2 +- entry_test.go | 4 +-- global_test.go | 8 +++--- lfu_test.go | 6 ++--- load_test.go | 4 +-- lru_test.go | 6 ++--- option_test.go | 36 +++++++++++++-------------- pkg/clock/clock_test.go | 4 +-- pkg/heap/heap_test.go | 4 +-- pkg/singleflight/singleflight_test.go | 8 +++--- pkg/task/task_test.go | 2 +- report_test.go | 20 +++++++-------- sharding_test.go | 4 +-- standard_test.go | 4 +-- 21 files changed, 76 insertions(+), 66 deletions(-) diff --git a/FUTURE.md b/FUTURE.md index 28eaeff..6bbb58e 100644 --- a/FUTURE.md +++ b/FUTURE.md @@ -1,11 +1,14 @@ ## ✒ 未来版本的新特性 (Features in future versions) +### v0.6.x + +* [ ] 梳理代码,优化代码风格,精简部分代码和注释 +* [ ] 完善监控上报器,提供更多缓存信息查询的方法 + ### v0.5.x * [ ] ~~提供一个清空并设置全量值的方法,方便定时数据的全量替换~~ 目前还找不到一个合适的设计去加入这个功能,并且也不是非常刚需,通过业务手段可以处理,所以先不加 -* [ ] 完善监控上报器,提供更多缓存信息查询的方法 -* [ ] 梳理代码,优化代码风格 ### v0.4.x diff --git a/HISTORY.md b/HISTORY.md index 57da66d..d3d0dbe 100644 --- a/HISTORY.md +++ b/HISTORY.md @@ -1,5 +1,12 @@ ## ✒ 历史版本的特性介绍 (Features in old versions) +### v0.6.0-alpha + +> 此版本发布于 2024-01-13 + +* 受小徒弟的建议,进行 Loader 代码的调整 +* 把 cache 结构去掉,精简这部分设计 + ### v0.5.0 > 此版本发布于 2023-11-30 diff --git a/README.en.md b/README.en.md index 128fffd..a835b6b 100644 --- a/README.en.md +++ b/README.en.md @@ -2,7 +2,7 @@ [![Go Doc](_icons/godoc.svg)](https://pkg.go.dev/github.com/FishGoddess/cachego) [![License](_icons/license.svg)](https://www.apache.org/licenses/LICENSE-2.0.html) -[![License](_icons/coverage.svg)](_icons/coverage.svg) +[![Coverage](_icons/coverage.svg)](_icons/coverage.svg) ![Test](https://github.com/FishGoddess/cachego/actions/workflows/test.yml/badge.svg) **cachego** is an api friendly memory-based cache for [GoLang](https://golang.org) applications. diff --git a/README.md b/README.md index 32c681e..8a07100 100644 --- a/README.md +++ b/README.md @@ -2,7 +2,7 @@ [![Go Doc](_icons/godoc.svg)](https://pkg.go.dev/github.com/FishGoddess/cachego) [![License](_icons/license.svg)](https://www.apache.org/licenses/LICENSE-2.0.html) -[![License](_icons/coverage.svg)](_icons/coverage.svg) +[![Coverage](_icons/coverage.svg)](_icons/coverage.svg) ![Test](https://github.com/FishGoddess/cachego/actions/workflows/test.yml/badge.svg) **cachego** 是一个拥有分片机制的轻量级内存缓存库,API 友好,支持多种数据淘汰机制,可以应用于所有的 [GoLang](https://golang.org) 应用程序中。 diff --git a/_icons/coverage.svg b/_icons/coverage.svg index be8453f..68ffdc7 100644 --- a/_icons/coverage.svg +++ b/_icons/coverage.svg @@ -10,7 +10,7 @@ coverage coverage - 98% - 98% + 97% + 97% \ No newline at end of file diff --git a/cache_test.go b/cache_test.go index 4172fbb..7abd5cc 100644 --- a/cache_test.go +++ b/cache_test.go @@ -252,7 +252,7 @@ func testCacheImplement(t *testing.T, cache Cache) { } } -// go test -v -cover=^TestNewCache$ +// go test -v -cover -count=1 -test.cpu=1=^TestNewCache$ func TestNewCache(t *testing.T) { cache := NewCache() @@ -296,7 +296,7 @@ func TestNewCache(t *testing.T) { cache = NewCache(WithLRU(0)) } -// go test -v -cover=^TestNewCacheWithReport$ +// go test -v -cover -count=1 -test.cpu=1=^TestNewCacheWithReport$ func TestNewCacheWithReport(t *testing.T) { cache, reporter := NewCacheWithReport() @@ -314,7 +314,7 @@ func TestNewCacheWithReport(t *testing.T) { } } -// go test -v -cover=^TestRunGCTask$ +// go test -v -cover -count=1 -test.cpu=1=^TestRunGCTask$ func TestRunGCTask(t *testing.T) { cache := new(testCache) diff --git a/cache_type_test.go b/cache_type_test.go index fe82ce1..fba7442 100644 --- a/cache_type_test.go +++ b/cache_type_test.go @@ -16,7 +16,7 @@ package cachego import "testing" -// go test -v -cover -run=^TestCacheType$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestCacheType$ func TestCacheType(t *testing.T) { if standard.String() != string(standard) { t.Fatalf("standard.String() %s is wrong", standard.String()) diff --git a/config_test.go b/config_test.go index 5758796..6491aa5 100644 --- a/config_test.go +++ b/config_test.go @@ -87,7 +87,7 @@ func isConfigEquals(conf1 *config, conf2 *config) bool { return true } -// go test -v -cover -run=^TestApplyOptions$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestApplyOptions$ func TestApplyOptions(t *testing.T) { got := &config{ shardings: 0, diff --git a/entry_test.go b/entry_test.go index dbcec72..ed3e08d 100644 --- a/entry_test.go +++ b/entry_test.go @@ -24,7 +24,7 @@ const ( testDurationGap = 10 * time.Microsecond ) -// go test -v -cover -run=^TestNewEntry$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestNewEntry$ func TestNewEntry(t *testing.T) { e := newEntry("key", "value", 0, now) @@ -69,7 +69,7 @@ func TestNewEntry(t *testing.T) { } } -// go test -v -cover -run=^TestEntrySetup$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestEntrySetup$ func TestEntrySetup(t *testing.T) { e := newEntry("key", "value", 0, now) diff --git a/global_test.go b/global_test.go index 9e4ebad..9a113b3 100644 --- a/global_test.go +++ b/global_test.go @@ -29,7 +29,7 @@ func BenchmarkHash(b *testing.B) { } } -// go test -v -cover -run=^TestHash$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestHash$ func TestHash(t *testing.T) { hash := hash("test") if hash < 0 { @@ -37,7 +37,7 @@ func TestHash(t *testing.T) { } } -// go test -v -cover -run=^TestNow$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestNow$ func TestNow(t *testing.T) { got := now() expect := time.Now().UnixNano() @@ -47,7 +47,7 @@ func TestNow(t *testing.T) { } } -// go test -v -cover -run=^TestSetMapInitialCap$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestSetMapInitialCap$ func TestSetMapInitialCap(t *testing.T) { oldInitialCap := mapInitialCap @@ -67,7 +67,7 @@ func TestSetMapInitialCap(t *testing.T) { } } -// go test -v -cover -run=^TestSetSliceInitialCap$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestSetSliceInitialCap$ func TestSetSliceInitialCap(t *testing.T) { oldInitialCap := sliceInitialCap diff --git a/lfu_test.go b/lfu_test.go index b4359e5..edaaf51 100644 --- a/lfu_test.go +++ b/lfu_test.go @@ -29,13 +29,13 @@ func newTestLFUCache() *lfuCache { return newLFUCache(conf).(*lfuCache) } -// go test -v -cover -run=^TestLFUCache$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestLFUCache$ func TestLFUCache(t *testing.T) { cache := newTestLFUCache() testCacheImplement(t, cache) } -// go test -v -cover -run=^TestLFUCacheEvict$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestLFUCacheEvict$ func TestLFUCacheEvict(t *testing.T) { cache := newTestLFUCache() @@ -85,7 +85,7 @@ func TestLFUCacheEvict(t *testing.T) { } } -// go test -v -cover -run=^TestLFUCacheEvictSimulate$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestLFUCacheEvictSimulate$ func TestLFUCacheEvictSimulate(t *testing.T) { cache := newTestLFUCache() diff --git a/load_test.go b/load_test.go index b0948dd..92fbf09 100644 --- a/load_test.go +++ b/load_test.go @@ -67,7 +67,7 @@ func (tlc *testLoadCache) Load(key string, ttl time.Duration, load func() (value return tlc.loader.Load(key, ttl, load) } -// go test -v -cover -run=^TestNewLoader$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestNewLoader$ func TestNewLoader(t *testing.T) { loader := NewLoader(false) if loader.group != nil { @@ -80,7 +80,7 @@ func TestNewLoader(t *testing.T) { } } -// go test -v -cover -run=^TestLoaderLoad$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestLoaderLoad$ func TestLoaderLoad(t *testing.T) { cache := newTestLoadCache(false) loadCount := 0 diff --git a/lru_test.go b/lru_test.go index aa9dd82..0060ddf 100644 --- a/lru_test.go +++ b/lru_test.go @@ -28,13 +28,13 @@ func newTestLRUCache() *lruCache { return newLRUCache(conf).(*lruCache) } -// go test -v -cover -run=^TestLRUCache$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestLRUCache$ func TestLRUCache(t *testing.T) { cache := newTestLRUCache() testCacheImplement(t, cache) } -// go test -v -cover -run=^TestLRUCacheEvict$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestLRUCacheEvict$ func TestLRUCacheEvict(t *testing.T) { cache := newTestLRUCache() @@ -74,7 +74,7 @@ func TestLRUCacheEvict(t *testing.T) { } } -// go test -v -cover -run=^TestLRUCacheEvictSimulate$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestLRUCacheEvictSimulate$ func TestLRUCacheEvictSimulate(t *testing.T) { cache := newTestLRUCache() diff --git a/option_test.go b/option_test.go index 5417b65..5a64b8e 100644 --- a/option_test.go +++ b/option_test.go @@ -19,7 +19,7 @@ import ( "time" ) -// go test -v -cover -run=^TestWithCacheName$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithCacheName$ func TestWithCacheName(t *testing.T) { got := &config{cacheName: ""} expect := &config{cacheName: "-"} @@ -30,7 +30,7 @@ func TestWithCacheName(t *testing.T) { } } -// go test -v -cover -run=^TestWithLRU$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithLRU$ func TestWithLRU(t *testing.T) { got := &config{cacheType: standard, maxEntries: 0} expect := &config{cacheType: lru, maxEntries: 666} @@ -41,7 +41,7 @@ func TestWithLRU(t *testing.T) { } } -// go test -v -cover -run=^TestWithLFU$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithLFU$ func TestWithLFU(t *testing.T) { got := &config{cacheType: standard, maxEntries: 0} expect := &config{cacheType: lfu, maxEntries: 999} @@ -52,7 +52,7 @@ func TestWithLFU(t *testing.T) { } } -// go test -v -cover -run=^TestWithShardings$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithShardings$ func TestWithShardings(t *testing.T) { got := &config{shardings: 0} expect := &config{shardings: 1024} @@ -63,7 +63,7 @@ func TestWithShardings(t *testing.T) { } } -// go test -v -cover -run=^TestWithDisableSingleflight$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithDisableSingleflight$ func TestWithDisableSingleflight(t *testing.T) { got := &config{singleflight: true} expect := &config{singleflight: false} @@ -74,7 +74,7 @@ func TestWithDisableSingleflight(t *testing.T) { } } -// go test -v -cover -run=^TestWithGC$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithGC$ func TestWithGC(t *testing.T) { got := &config{gcDuration: 0} expect := &config{gcDuration: 1024} @@ -85,7 +85,7 @@ func TestWithGC(t *testing.T) { } } -// go test -v -cover -run=^TestWithMaxScans$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithMaxScans$ func TestWithMaxScans(t *testing.T) { got := &config{maxScans: 0} expect := &config{maxScans: 1024} @@ -96,7 +96,7 @@ func TestWithMaxScans(t *testing.T) { } } -// go test -v -cover -run=^TestWithMaxEntries$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithMaxEntries$ func TestWithMaxEntries(t *testing.T) { got := &config{maxEntries: 0} expect := &config{maxEntries: 1024} @@ -107,7 +107,7 @@ func TestWithMaxEntries(t *testing.T) { } } -// go test -v -cover -run=^TestWithNow$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithNow$ func TestWithNow(t *testing.T) { now := func() int64 { return 0 @@ -122,7 +122,7 @@ func TestWithNow(t *testing.T) { } } -// go test -v -cover -run=^TestWithHash$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithHash$ func TestWithHash(t *testing.T) { hash := func(key string) int { return 0 @@ -137,7 +137,7 @@ func TestWithHash(t *testing.T) { } } -// go test -v -cover -run=^TestWithRecordMissed$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithRecordMissed$ func TestWithRecordMissed(t *testing.T) { got := &config{recordMissed: false} expect := &config{recordMissed: true} @@ -148,7 +148,7 @@ func TestWithRecordMissed(t *testing.T) { } } -// go test -v -cover -run=^TestWithRecordHit$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithRecordHit$ func TestWithRecordHit(t *testing.T) { got := &config{recordHit: false} expect := &config{recordHit: true} @@ -159,7 +159,7 @@ func TestWithRecordHit(t *testing.T) { } } -// go test -v -cover -run=^TestWithRecordGC$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithRecordGC$ func TestWithRecordGC(t *testing.T) { got := &config{recordGC: false} expect := &config{recordGC: true} @@ -170,7 +170,7 @@ func TestWithRecordGC(t *testing.T) { } } -// go test -v -cover -run=^TestWithRecordLoad$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithRecordLoad$ func TestWithRecordLoad(t *testing.T) { got := &config{recordLoad: false} expect := &config{recordLoad: true} @@ -181,7 +181,7 @@ func TestWithRecordLoad(t *testing.T) { } } -// go test -v -cover -run=^TestWithReportMissed$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithReportMissed$ func TestWithReportMissed(t *testing.T) { reportMissed := func(reporter *Reporter, key string) {} @@ -194,7 +194,7 @@ func TestWithReportMissed(t *testing.T) { } } -// go test -v -cover -run=^TestWithReportHit$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithReportHit$ func TestWithReportHit(t *testing.T) { reportHit := func(reporter *Reporter, key string, value interface{}) {} @@ -207,7 +207,7 @@ func TestWithReportHit(t *testing.T) { } } -// go test -v -cover -run=^TestWithReportGC$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithReportGC$ func TestWithReportGC(t *testing.T) { reportGC := func(reporter *Reporter, cost time.Duration, cleans int) {} @@ -220,7 +220,7 @@ func TestWithReportGC(t *testing.T) { } } -// go test -v -cover -run=^TestWithReportLoad$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestWithReportLoad$ func TestWithReportLoad(t *testing.T) { reportLoad := func(reporter *Reporter, key string, value interface{}, ttl time.Duration, err error) {} diff --git a/pkg/clock/clock_test.go b/pkg/clock/clock_test.go index f44b95e..aa70164 100644 --- a/pkg/clock/clock_test.go +++ b/pkg/clock/clock_test.go @@ -43,7 +43,7 @@ func BenchmarkClockNow(b *testing.B) { } } -// go test -v -cover -run=^TestNew$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestNew$ func TestNew(t *testing.T) { var clocks []*Clock @@ -58,7 +58,7 @@ func TestNew(t *testing.T) { } } -// go test -v -cover -run=^TestClock$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestClock$ func TestClock(t *testing.T) { testClock := New() diff --git a/pkg/heap/heap_test.go b/pkg/heap/heap_test.go index f496fcd..b74556b 100644 --- a/pkg/heap/heap_test.go +++ b/pkg/heap/heap_test.go @@ -33,7 +33,7 @@ func newTestData(count int) []int { return data } -// go test -v -cover -run=^TestItem$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestItem$ func TestItem(t *testing.T) { heap := New(64) @@ -107,7 +107,7 @@ func TestItem(t *testing.T) { } } -// go test -v -cover -run=^TestHeap$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestHeap$ func TestHeap(t *testing.T) { data := newTestData(10) t.Log(data) diff --git a/pkg/singleflight/singleflight_test.go b/pkg/singleflight/singleflight_test.go index 0367194..7daa777 100644 --- a/pkg/singleflight/singleflight_test.go +++ b/pkg/singleflight/singleflight_test.go @@ -55,13 +55,13 @@ func testGroupCall(t *testing.T, group *Group, concurrency int) { wg.Wait() } -// go test -v -cover -run=^TestGroupCall$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestGroupCall$ func TestGroupCall(t *testing.T) { group := NewGroup(128) testGroupCall(t, group, 100000) } -// go test -v -cover -run=^TestGroupCallMultiKey$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestGroupCallMultiKey$ func TestGroupCallMultiKey(t *testing.T) { group := NewGroup(128) @@ -78,7 +78,7 @@ func TestGroupCallMultiKey(t *testing.T) { wg.Wait() } -// go test -v -cover -run=^TestGroupDelete$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestGroupDelete$ func TestGroupDelete(t *testing.T) { group := NewGroup(128) @@ -114,7 +114,7 @@ func TestGroupDelete(t *testing.T) { } } -// go test -v -cover -run=^TestGroupReset$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestGroupReset$ func TestGroupReset(t *testing.T) { group := NewGroup(128) diff --git a/pkg/task/task_test.go b/pkg/task/task_test.go index c074c04..3b5fc3d 100644 --- a/pkg/task/task_test.go +++ b/pkg/task/task_test.go @@ -27,7 +27,7 @@ type testEntry struct { value string } -// go test -v -cover -run=^TestTickerTaskRun$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestTickerTaskRun$ func TestTickerTaskRun(t *testing.T) { before := testEntry{key: "before_key", value: "before_value"} fn := testEntry{key: "task_key", value: "task_value"} diff --git a/report_test.go b/report_test.go index 607a8cf..9619ed0 100644 --- a/report_test.go +++ b/report_test.go @@ -39,13 +39,13 @@ func newTestReportableCache() (*reportableCache, *Reporter) { return cache.(*reportableCache), reporter } -// go test -v -cover -run=^TestReportableCache$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestReportableCache$ func TestReportableCache(t *testing.T) { cache, _ := newTestReportableCache() testCacheImplement(t, cache) } -// go test -v -cover -run=^TestReportableCacheReportMissed$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestReportableCacheReportMissed$ func TestReportableCacheReportMissed(t *testing.T) { cache, reporter := newTestReportableCache() cache.Set("key", 666, NoTTL) @@ -80,7 +80,7 @@ func TestReportableCacheReportMissed(t *testing.T) { } } -// go test -v -cover -run=^TestReportableCacheReportHit$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestReportableCacheReportHit$ func TestReportableCacheReportHit(t *testing.T) { cache, reporter := newTestReportableCache() cache.Set("key", 666, NoTTL) @@ -119,7 +119,7 @@ func TestReportableCacheReportHit(t *testing.T) { } } -// go test -v -cover -run=^TestReportableCacheReportGC$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestReportableCacheReportGC$ func TestReportableCacheReportGC(t *testing.T) { cache, reporter := newTestReportableCache() cache.Set("key1", 1, time.Millisecond) @@ -160,7 +160,7 @@ func TestReportableCacheReportGC(t *testing.T) { } } -// go test -v -cover -run=^TestReportableCacheReportLoad$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestReportableCacheReportLoad$ func TestReportableCacheReportLoad(t *testing.T) { cache, reporter := newTestReportableCache() @@ -209,7 +209,7 @@ func TestReportableCacheReportLoad(t *testing.T) { } } -// go test -v -cover -run=^TestReporterCacheName$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestReporterCacheName$ func TestReporterCacheName(t *testing.T) { _, reporter := newTestReportableCache() if reporter.CacheName() != reporter.conf.cacheName { @@ -221,7 +221,7 @@ func TestReporterCacheName(t *testing.T) { } } -// go test -v -cover -run=^TestReporterCacheType$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestReporterCacheType$ func TestReporterCacheType(t *testing.T) { _, reporter := newTestReportableCache() if reporter.CacheType() != reporter.conf.cacheType { @@ -233,7 +233,7 @@ func TestReporterCacheType(t *testing.T) { } } -// go test -v -cover -run=^TestReporterCacheShardings$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestReporterCacheShardings$ func TestReporterCacheShardings(t *testing.T) { _, reporter := newTestReportableCache() if reporter.CacheShardings() != reporter.conf.shardings { @@ -245,7 +245,7 @@ func TestReporterCacheShardings(t *testing.T) { } } -// go test -v -cover -run=^TestReporterCacheGC$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestReporterCacheGC$ func TestReporterCacheGC(t *testing.T) { _, reporter := newTestReportableCache() if reporter.CacheGC() != reporter.conf.gcDuration { @@ -257,7 +257,7 @@ func TestReporterCacheGC(t *testing.T) { } } -// go test -v -cover -run=^TestReporterCacheSize$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestReporterCacheSize$ func TestReporterCacheSize(t *testing.T) { cache, reporter := newTestReportableCache() cache.Set("key1", 1, time.Millisecond) diff --git a/sharding_test.go b/sharding_test.go index 86f3f1f..71c7d22 100644 --- a/sharding_test.go +++ b/sharding_test.go @@ -30,13 +30,13 @@ func newTestShardingCache() *shardingCache { return newShardingCache(conf, newStandardCache).(*shardingCache) } -// go test -v -cover -run=^TestShardingCache$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestShardingCache$ func TestShardingCache(t *testing.T) { cache := newTestShardingCache() testCacheImplement(t, cache) } -// go test -v -cover -run=^TestShardingCacheIndex$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestShardingCacheIndex$ func TestShardingCacheIndex(t *testing.T) { cache := newTestShardingCache() diff --git a/standard_test.go b/standard_test.go index 5bb82c2..3af3b85 100644 --- a/standard_test.go +++ b/standard_test.go @@ -27,13 +27,13 @@ func newTestStandardCache() *standardCache { return newStandardCache(conf).(*standardCache) } -// go test -v -cover -run=^TestStandardCache$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestStandardCache$ func TestStandardCache(t *testing.T) { cache := newTestStandardCache() testCacheImplement(t, cache) } -// go test -v -cover -run=^TestStandardCacheEvict$ +// go test -v -cover -count=1 -test.cpu=1 -run=^TestStandardCacheEvict$ func TestStandardCacheEvict(t *testing.T) { cache := newTestStandardCache() From a30d1267937c891805cc60033e91bdeeed11e6a3 Mon Sep 17 00:00:00 2001 From: FishGoddess <1149062639@qq.com> Date: Sat, 13 Jan 2024 20:48:05 +0800 Subject: [PATCH 3/6] =?UTF-8?q?=E8=B0=83=E6=95=B4=E6=A1=88=E4=BE=8B?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- _examples/task.go | 7 +------ doc.go | 7 +------ 2 files changed, 2 insertions(+), 12 deletions(-) diff --git a/_examples/task.go b/_examples/task.go index c9c27b0..db814ad 100644 --- a/_examples/task.go +++ b/_examples/task.go @@ -54,10 +54,5 @@ func main() { // Duration is the duration between two loop of fn, optional. // Run will start a new goroutine and run the task loop. // The task will stop if context is done. - task.New(printContextValue). - Before(beforePrint). - After(afterPrint). - Context(ctx). - Duration(time.Second). - Run() + task.New(printContextValue).Before(beforePrint).After(afterPrint).Context(ctx).Duration(time.Second).Run() } diff --git a/doc.go b/doc.go index 819a092..a9334bc 100644 --- a/doc.go +++ b/doc.go @@ -437,12 +437,7 @@ Package cachego provides an easy way to use foundation for your caching operatio // Duration is the duration between two loop of fn, optional. // Run will start a new goroutine and run the task loop. // The task will stop if context is done. - task.New(printContextValue). - Before(beforePrint). - After(afterPrint). - Context(ctx). - Duration(time.Second). - Run() + task.New(printContextValue).Before(beforePrint).After(afterPrint).Context(ctx).Duration(time.Second).Run() 10. clock: From fd7f297466381b289a683f6b985ade7a4b583fe0 Mon Sep 17 00:00:00 2001 From: FishGoddess <1149062639@qq.com> Date: Sat, 13 Jan 2024 20:49:26 +0800 Subject: [PATCH 4/6] =?UTF-8?q?=E8=B0=83=E6=95=B4=E6=A1=88=E4=BE=8B?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- HISTORY.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/HISTORY.md b/HISTORY.md index d3d0dbe..badde72 100644 --- a/HISTORY.md +++ b/HISTORY.md @@ -4,7 +4,7 @@ > 此版本发布于 2024-01-13 -* 受小徒弟的建议,进行 Loader 代码的调整 +* 受小徒弟的灵感激发,进行 Loader 代码的调整 * 把 cache 结构去掉,精简这部分设计 ### v0.5.0 From 5a62ec8c36040b42bab784b0e6bc0134a5a43dbc Mon Sep 17 00:00:00 2001 From: FishGoddess <1149062639@qq.com> Date: Sat, 13 Jan 2024 20:55:29 +0800 Subject: [PATCH 5/6] =?UTF-8?q?=E5=B0=86=20Loader=20=E6=94=B9=E6=88=90=20l?= =?UTF-8?q?oader=20=E7=A7=81=E6=9C=89=E7=9A=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- HISTORY.md | 2 +- cache_test.go | 2 +- lfu.go | 2 +- load.go | 12 ++++++------ load_test.go | 2 +- lru.go | 2 +- standard.go | 2 +- 7 files changed, 12 insertions(+), 12 deletions(-) diff --git a/HISTORY.md b/HISTORY.md index badde72..6d48994 100644 --- a/HISTORY.md +++ b/HISTORY.md @@ -4,7 +4,7 @@ > 此版本发布于 2024-01-13 -* 受小徒弟的灵感激发,进行 Loader 代码的调整 +* 受小徒弟的灵感激发,进行 loader 代码的调整 * 把 cache 结构去掉,精简这部分设计 ### v0.5.0 diff --git a/cache_test.go b/cache_test.go index 7abd5cc..72413e7 100644 --- a/cache_test.go +++ b/cache_test.go @@ -27,7 +27,7 @@ const ( type testCache struct { *config - loader *Loader + loader *loader count int32 } diff --git a/lfu.go b/lfu.go index 7a04e57..d4fd977 100644 --- a/lfu.go +++ b/lfu.go @@ -28,7 +28,7 @@ type lfuCache struct { itemHeap *heap.Heap lock sync.RWMutex - loader *Loader + loader *loader } func newLFUCache(conf *config) Cache { diff --git a/load.go b/load.go index 5af0370..a349ea1 100644 --- a/load.go +++ b/load.go @@ -21,15 +21,15 @@ import ( flight "github.com/FishGoddess/cachego/pkg/singleflight" ) -// Loader loads values from somewhere. -type Loader struct { +// loader loads values from somewhere. +type loader struct { group *flight.Group } // NewLoader creates a loader. // It also creates a singleflight group to call load if singleflight is true. -func NewLoader(singleflight bool) *Loader { - loader := new(Loader) +func NewLoader(singleflight bool) *loader { + loader := new(loader) if singleflight { loader.group = flight.NewGroup(mapInitialCap) @@ -39,7 +39,7 @@ func NewLoader(singleflight bool) *Loader { } // Load loads a value of key with ttl and returns an error if failed. -func (l *Loader) Load(key string, ttl time.Duration, load func() (value interface{}, err error)) (value interface{}, err error) { +func (l *loader) Load(key string, ttl time.Duration, load func() (value interface{}, err error)) (value interface{}, err error) { if load == nil { return nil, errors.New("cachego: load function is nil in loader") } @@ -52,7 +52,7 @@ func (l *Loader) Load(key string, ttl time.Duration, load func() (value interfac } // Reset resets loader to initial status which is like a new loader. -func (l *Loader) Reset() { +func (l *loader) Reset() { if l.group != nil { l.group.Reset() } diff --git a/load_test.go b/load_test.go index 92fbf09..21295eb 100644 --- a/load_test.go +++ b/load_test.go @@ -26,7 +26,7 @@ type testLoadCache struct { value interface{} ttl time.Duration - loader *Loader + loader *loader } func newTestLoadCache(singleflight bool) Cache { diff --git a/lru.go b/lru.go index 40bef21..d639bed 100644 --- a/lru.go +++ b/lru.go @@ -27,7 +27,7 @@ type lruCache struct { elementList *list.List lock sync.RWMutex - loader *Loader + loader *loader } func newLRUCache(conf *config) Cache { diff --git a/standard.go b/standard.go index b2fa599..f99a8ad 100644 --- a/standard.go +++ b/standard.go @@ -25,7 +25,7 @@ type standardCache struct { entries map[string]*entry lock sync.RWMutex - loader *Loader + loader *loader } func newStandardCache(conf *config) Cache { From 0426d612bbe51324ae505d4919da8443cd9e2aab Mon Sep 17 00:00:00 2001 From: FishGoddess <1149062639@qq.com> Date: Sat, 13 Jan 2024 21:03:37 +0800 Subject: [PATCH 6/6] =?UTF-8?q?=E5=B0=86=20NewLoader=20=E6=94=B9=E6=88=90?= =?UTF-8?q?=E7=A7=81=E6=9C=89=E7=9A=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- lfu.go | 2 +- load.go | 4 ++-- load_test.go | 6 +++--- lru.go | 2 +- standard.go | 2 +- 5 files changed, 8 insertions(+), 8 deletions(-) diff --git a/lfu.go b/lfu.go index d4fd977..1705f4b 100644 --- a/lfu.go +++ b/lfu.go @@ -40,7 +40,7 @@ func newLFUCache(conf *config) Cache { config: conf, itemMap: make(map[string]*heap.Item, mapInitialCap), itemHeap: heap.New(sliceInitialCap), - loader: NewLoader(conf.singleflight), + loader: newLoader(conf.singleflight), } return cache diff --git a/load.go b/load.go index a349ea1..b5e3f76 100644 --- a/load.go +++ b/load.go @@ -26,9 +26,9 @@ type loader struct { group *flight.Group } -// NewLoader creates a loader. +// newLoader creates a loader. // It also creates a singleflight group to call load if singleflight is true. -func NewLoader(singleflight bool) *loader { +func newLoader(singleflight bool) *loader { loader := new(loader) if singleflight { diff --git a/load_test.go b/load_test.go index 21295eb..0b473f9 100644 --- a/load_test.go +++ b/load_test.go @@ -31,7 +31,7 @@ type testLoadCache struct { func newTestLoadCache(singleflight bool) Cache { cache := &testLoadCache{ - loader: NewLoader(singleflight), + loader: newLoader(singleflight), } return cache @@ -69,12 +69,12 @@ func (tlc *testLoadCache) Load(key string, ttl time.Duration, load func() (value // go test -v -cover -count=1 -test.cpu=1 -run=^TestNewLoader$ func TestNewLoader(t *testing.T) { - loader := NewLoader(false) + loader := newLoader(false) if loader.group != nil { t.Fatalf("loader.group %+v != nil", loader.group) } - loader = NewLoader(true) + loader = newLoader(true) if loader.group == nil { t.Fatal("loader.group == nil") } diff --git a/lru.go b/lru.go index d639bed..f023c77 100644 --- a/lru.go +++ b/lru.go @@ -39,7 +39,7 @@ func newLRUCache(conf *config) Cache { config: conf, elementMap: make(map[string]*list.Element, mapInitialCap), elementList: list.New(), - loader: NewLoader(conf.singleflight), + loader: newLoader(conf.singleflight), } return cache diff --git a/standard.go b/standard.go index f99a8ad..83dd0ad 100644 --- a/standard.go +++ b/standard.go @@ -32,7 +32,7 @@ func newStandardCache(conf *config) Cache { cache := &standardCache{ config: conf, entries: make(map[string]*entry, mapInitialCap), - loader: NewLoader(conf.singleflight), + loader: newLoader(conf.singleflight), } return cache