1
0
mirror of https://github.com/gogf/gf.git synced 2025-04-05 03:05:05 +08:00
gf/os/gcache/gcache_adapter_memory_data.go

215 lines
5.4 KiB
Go

// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
//
// This Source Code Form is subject to the terms of the MIT License.
// If a copy of the MIT was not distributed with this file,
// You can obtain one at https://github.com/gogf/gf.
package gcache
import (
"context"
"sync"
"time"
"github.com/gogf/gf/v2/os/gtime"
)
type memoryData struct {
mu sync.RWMutex // dataMu ensures the concurrent safety of underlying data map.
data map[interface{}]memoryDataItem // data is the underlying cache data which is stored in a hash table.
}
// memoryDataItem holds the internal cache item data.
type memoryDataItem struct {
v interface{} // Value.
e int64 // Expire timestamp in milliseconds.
}
func newMemoryData() *memoryData {
return &memoryData{
data: make(map[interface{}]memoryDataItem),
}
}
// Update updates the value of `key` without changing its expiration and returns the old value.
// The returned value `exist` is false if the `key` does not exist in the cache.
//
// It deletes the `key` if given `value` is nil.
// It does nothing if `key` does not exist in the cache.
func (d *memoryData) Update(key interface{}, value interface{}) (oldValue interface{}, exist bool, err error) {
d.mu.Lock()
defer d.mu.Unlock()
if item, ok := d.data[key]; ok {
d.data[key] = memoryDataItem{
v: value,
e: item.e,
}
return item.v, true, nil
}
return nil, false, nil
}
// UpdateExpire updates the expiration of `key` and returns the old expiration duration value.
//
// It returns -1 and does nothing if the `key` does not exist in the cache.
// It deletes the `key` if `duration` < 0.
func (d *memoryData) UpdateExpire(key interface{}, expireTime int64) (oldDuration time.Duration, err error) {
d.mu.Lock()
defer d.mu.Unlock()
if item, ok := d.data[key]; ok {
d.data[key] = memoryDataItem{
v: item.v,
e: expireTime,
}
return time.Duration(item.e-gtime.TimestampMilli()) * time.Millisecond, nil
}
return -1, nil
}
// Remove deletes the one or more keys from cache, and returns its value.
// If multiple keys are given, it returns the value of the deleted last item.
func (d *memoryData) Remove(keys ...interface{}) (removedKeys []interface{}, value interface{}, err error) {
d.mu.Lock()
defer d.mu.Unlock()
removedKeys = make([]interface{}, 0)
for _, key := range keys {
item, ok := d.data[key]
if ok {
value = item.v
delete(d.data, key)
removedKeys = append(removedKeys, key)
}
}
return removedKeys, value, nil
}
// Data returns a copy of all key-value pairs in the cache as map type.
func (d *memoryData) Data() (map[interface{}]interface{}, error) {
d.mu.RLock()
defer d.mu.RUnlock()
var (
data = make(map[interface{}]interface{}, len(d.data))
nowMilli = gtime.TimestampMilli()
)
for k, v := range d.data {
if v.e > nowMilli {
data[k] = v.v
}
}
return data, nil
}
// Keys returns all keys in the cache as slice.
func (d *memoryData) Keys() ([]interface{}, error) {
d.mu.RLock()
defer d.mu.RUnlock()
var (
keys = make([]interface{}, 0, len(d.data))
nowMilli = gtime.TimestampMilli()
)
for k, v := range d.data {
if v.e > nowMilli {
keys = append(keys, k)
}
}
return keys, nil
}
// Values returns all values in the cache as slice.
func (d *memoryData) Values() ([]interface{}, error) {
d.mu.RLock()
defer d.mu.RUnlock()
var (
values = make([]interface{}, 0, len(d.data))
nowMilli = gtime.TimestampMilli()
)
for _, v := range d.data {
if v.e > nowMilli {
values = append(values, v.v)
}
}
return values, nil
}
// Size returns the size of the cache that not expired.
func (d *memoryData) Size() (size int, err error) {
d.mu.RLock()
defer d.mu.RUnlock()
var nowMilli = gtime.TimestampMilli()
for _, v := range d.data {
if v.e > nowMilli {
size++
}
}
return size, nil
}
// Clear clears all data of the cache.
// Note that this function is sensitive and should be carefully used.
func (d *memoryData) Clear() {
d.mu.Lock()
defer d.mu.Unlock()
d.data = make(map[interface{}]memoryDataItem)
}
func (d *memoryData) Get(key interface{}) (item memoryDataItem, ok bool) {
d.mu.RLock()
item, ok = d.data[key]
d.mu.RUnlock()
return
}
func (d *memoryData) Set(key interface{}, value memoryDataItem) {
d.mu.Lock()
d.data[key] = value
d.mu.Unlock()
}
// SetMap batch sets cache with key-value pairs by `data`, which is expired after `duration`.
//
// It does not expire if `duration` == 0.
// It deletes the keys of `data` if `duration` < 0 or given `value` is nil.
func (d *memoryData) SetMap(data map[interface{}]interface{}, expireTime int64) error {
d.mu.Lock()
for k, v := range data {
d.data[k] = memoryDataItem{
v: v,
e: expireTime,
}
}
d.mu.Unlock()
return nil
}
func (d *memoryData) SetWithLock(ctx context.Context, key interface{}, value interface{}, expireTimestamp int64) (interface{}, error) {
d.mu.Lock()
defer d.mu.Unlock()
var (
err error
)
if v, ok := d.data[key]; ok && !v.IsExpired() {
return v.v, nil
}
f, ok := value.(Func)
if !ok {
// Compatible with raw function value.
f, ok = value.(func(ctx context.Context) (value interface{}, err error))
}
if ok {
if value, err = f(ctx); err != nil {
return nil, err
}
if value == nil {
return nil, nil
}
}
d.data[key] = memoryDataItem{v: value, e: expireTimestamp}
return value, nil
}
func (d *memoryData) Delete(key interface{}) {
d.mu.Lock()
defer d.mu.Unlock()
delete(d.data, key)
}