mirror of
https://github.com/talent-plan/tinykv.git
synced 2025-02-07 18:20:19 +08:00
Signed-off-by: Connor <zbk602423539@gmail.com> Co-authored-by: Nick Cameron <nrc@ncameron.org> Co-authored-by: linning <linningde25@gmail.com> Co-authored-by: YangKeao <keao.yang@yahoo.com> Co-authored-by: andylokandy <andylokandy@hotmail.com> Co-authored-by: Iosmanthus Teng <myosmanthustree@gmail.com>
153 lines
2.9 KiB
Go
153 lines
2.9 KiB
Go
// Copyright 2017 PingCAP, Inc.
|
|
//
|
|
// 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,
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package cache
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/pingcap/log"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
type ttlCacheItem struct {
|
|
value interface{}
|
|
expire time.Time
|
|
}
|
|
|
|
// TTL is a cache that assigns TTL(Time-To-Live) for each items.
|
|
type TTL struct {
|
|
sync.RWMutex
|
|
ctx context.Context
|
|
|
|
items map[uint64]ttlCacheItem
|
|
ttl time.Duration
|
|
gcInterval time.Duration
|
|
}
|
|
|
|
// NewTTL returns a new TTL cache.
|
|
func NewTTL(ctx context.Context, gcInterval time.Duration, ttl time.Duration) *TTL {
|
|
c := &TTL{
|
|
ctx: ctx,
|
|
items: make(map[uint64]ttlCacheItem),
|
|
ttl: ttl,
|
|
gcInterval: gcInterval,
|
|
}
|
|
|
|
go c.doGC()
|
|
return c
|
|
}
|
|
|
|
// Put puts an item into cache.
|
|
func (c *TTL) Put(key uint64, value interface{}) {
|
|
c.PutWithTTL(key, value, c.ttl)
|
|
}
|
|
|
|
// PutWithTTL puts an item into cache with specified TTL.
|
|
func (c *TTL) PutWithTTL(key uint64, value interface{}, ttl time.Duration) {
|
|
c.Lock()
|
|
defer c.Unlock()
|
|
|
|
c.items[key] = ttlCacheItem{
|
|
value: value,
|
|
expire: time.Now().Add(ttl),
|
|
}
|
|
}
|
|
|
|
// Get retrives an item from cache.
|
|
func (c *TTL) Get(key uint64) (interface{}, bool) {
|
|
c.RLock()
|
|
defer c.RUnlock()
|
|
|
|
item, ok := c.items[key]
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
|
|
if item.expire.Before(time.Now()) {
|
|
return nil, false
|
|
}
|
|
|
|
return item.value, true
|
|
}
|
|
|
|
// Remove eliminates an item from cache.
|
|
func (c *TTL) Remove(key uint64) {
|
|
c.Lock()
|
|
defer c.Unlock()
|
|
|
|
delete(c.items, key)
|
|
}
|
|
|
|
// Len returns current cache size.
|
|
func (c *TTL) Len() int {
|
|
c.RLock()
|
|
defer c.RUnlock()
|
|
|
|
return len(c.items)
|
|
}
|
|
|
|
// Clear removes all items in the ttl cache.
|
|
func (c *TTL) Clear() {
|
|
c.Lock()
|
|
defer c.Unlock()
|
|
|
|
for k := range c.items {
|
|
delete(c.items, k)
|
|
}
|
|
}
|
|
|
|
func (c *TTL) doGC() {
|
|
ticker := time.NewTicker(c.gcInterval)
|
|
defer ticker.Stop()
|
|
|
|
for {
|
|
select {
|
|
case <-ticker.C:
|
|
count := 0
|
|
now := time.Now()
|
|
c.Lock()
|
|
for key := range c.items {
|
|
if value, ok := c.items[key]; ok {
|
|
if value.expire.Before(now) {
|
|
count++
|
|
delete(c.items, key)
|
|
}
|
|
}
|
|
}
|
|
c.Unlock()
|
|
log.Debug("TTL GC items", zap.Int("count", count))
|
|
case <-c.ctx.Done():
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
// TTLUint64 is simple TTL saves only uint64s.
|
|
type TTLUint64 struct {
|
|
*TTL
|
|
}
|
|
|
|
// Put saves an ID in cache.
|
|
func (c *TTLUint64) Put(id uint64) {
|
|
c.TTL.Put(id, nil)
|
|
}
|
|
|
|
// Exists checks if an ID exists in cache.
|
|
func (c *TTLUint64) Exists(id uint64) bool {
|
|
_, ok := c.TTL.Get(id)
|
|
return ok
|
|
}
|