!62 增加清理事件

Merge pull request !62 from K-on/add-Clear-Event
This commit is contained in:
bandl 2021-10-11 12:40:42 +00:00 committed by Gitee
commit e826ff4569
11 changed files with 135 additions and 19 deletions

View File

@ -9,7 +9,7 @@ storage:
# clearSize and maxSize must be Int # clearSize and maxSize must be Int
lruCache: lruCache:
clearSize: "512MB" clearSize: "512mb"
maxSize: "1GB" maxSize: "1GB"
eventDriverSize: 2000 eventDriverSize: 2000
workTime: 1 workTime: 1

5
pkg/errorx/lru.go Normal file
View File

@ -0,0 +1,5 @@
package errorx
func LruNotWorkFuncEventErr() error {
return New("the event haven't work of function")
}

View File

@ -18,6 +18,7 @@ type EventWorkFunc func() (interface{}, error)
type DriverInterface interface { type DriverInterface interface {
Get() *Event Get() *Event
Put(event *Event) Put(event *Event)
GetLength() int
} }
type ProduceInterface interface { type ProduceInterface interface {

View File

@ -137,6 +137,10 @@ func (d *Driver) Put(event *Event) {
d.queue <- event d.queue <- event
} }
func (d *Driver) GetLength() int {
return len(d.queue)
}
// NewDriver 新建 Driver // NewDriver 新建 Driver
func NewDriver(maxSize int) DriverInterface { func NewDriver(maxSize int) DriverInterface {
return &Driver{ return &Driver{

31
pkg/lru/clean_work.go Normal file
View File

@ -0,0 +1,31 @@
package lru
import (
"context"
"gitee.com/timedb/wheatCache/pkg/event"
"time"
)
func (lru *SingleCache) cleanWork() {
cxt := context.Background()
for {
time.Sleep(2 * time.Second)
if lru.clearSize < lru.nowSize {
lruCleanEvent := event.NewEvent(CleanEventName)
lruCleanEvent.InitWaitEvent()
work := event.EventWorkFunc(func() (interface{}, error) {
err := lru.DelToClearSize()
return nil, err
})
lruCleanEvent.SetValue(WorkFuncEventKey, work)
lru.lruCleanProduce.Call(cxt, lruCleanEvent)
_, err := lruCleanEvent.StartWaitEvent(defaultWaitTime)
if err != nil {
//logx.With(cxt, ).Error("cleanWork err: %v", err)
}
}
}
}

View File

@ -4,6 +4,7 @@ import (
"gitee.com/timedb/wheatCache/pkg/proto" "gitee.com/timedb/wheatCache/pkg/proto"
"gitee.com/timedb/wheatCache/pkg/structure" "gitee.com/timedb/wheatCache/pkg/structure"
"sync" "sync"
"time"
) )
type SingleWorkFunc func() interface{} type SingleWorkFunc func() interface{}
@ -20,9 +21,12 @@ var (
) )
const ( const (
lruMaxSize = 1 * 1024 * 1024 * 1024 * 8 defaultLruMaxSize = 1 * 1024 * 1024 * 1024 * 8
lruClearSize = 0.5 * 1024 * 1024 * 1024 * 8 defaultLruClearSize = 0.5 * 1024 * 1024 * 1024 * 8
lruEventDriver = 2000 defaultLruEventDriver = 2000
)
const (
defaultWaitTime = 20 * time.Minute
) )
type CacheInterface interface { type CacheInterface interface {
@ -31,4 +35,5 @@ type CacheInterface interface {
Add(key *proto.BaseKey, val structure.KeyBaseInterface) Add(key *proto.BaseKey, val structure.KeyBaseInterface)
UpdateLruSize(length structure.UpdateLength) UpdateLruSize(length structure.UpdateLength)
DelByKey(key *proto.BaseKey) error DelByKey(key *proto.BaseKey) error
DelToClearSize() error
} }

View File

@ -18,12 +18,12 @@ type keyBaseValue struct {
} }
type SingleCache struct { type SingleCache struct {
maxsize int64 //最大的长度 maxsize int64 //最大的长度
clearSize int64 // 清理长度 clearSize int64 // 清理长度
nowSize int64 // 现在的长度 nowSize int64 // 现在的长度
li *list.List li *list.List
lruMap map[string]*list.Element lruMap map[string]*list.Element
lruMaxDiverSize int
lruDriver event.DriverInterface lruDriver event.DriverInterface
lruConsumer event.ConsumerInterface lruConsumer event.ConsumerInterface
lruCleanProduce event.ProduceInterface // 发送清理事件 lruCleanProduce event.ProduceInterface // 发送清理事件
@ -41,7 +41,7 @@ func cacheInit() (int64, int64, int) {
return 0, 0, 0 return 0, 0, 0
} }
if retMaxSize == 0 { if retMaxSize == 0 {
retMaxSize = lruMaxSize retMaxSize = defaultLruMaxSize
} }
clearSize := viper.GetString("lruCache.clearSize") clearSize := viper.GetString("lruCache.clearSize")
@ -50,12 +50,12 @@ func cacheInit() (int64, int64, int) {
return 0, 0, 0 return 0, 0, 0
} }
if retClearSize == 0 { if retClearSize == 0 {
retClearSize = lruClearSize retClearSize = defaultLruClearSize
} }
maxDriver := viper.GetInt("lruCache.eventDriverSize") maxDriver := viper.GetInt("lruCache.eventDriverSize")
if maxDriver == 0 { if maxDriver == 0 {
maxDriver = lruEventDriver maxDriver = defaultLruEventDriver
} }
return retMaxSize, retClearSize, maxDriver return retMaxSize, retClearSize, maxDriver
} }
@ -71,6 +71,7 @@ func NewLRUCache() *SingleCache {
nowSize: 0, nowSize: 0,
li: list.New(), li: list.New(),
lruMap: make(map[string]*list.Element), lruMap: make(map[string]*list.Element),
lruMaxDiverSize: maxDriverSize,
lruDriver: lruDriver, lruDriver: lruDriver,
lruConsumer: event.NewConsumer(lruDriver), lruConsumer: event.NewConsumer(lruDriver),
lruCleanProduce: event.NewProduce(lruDriver), lruCleanProduce: event.NewProduce(lruDriver),
@ -131,7 +132,7 @@ func (lru *SingleCache) Del() error {
} }
//DelByKey 根据key删除 //DelByKey 根据key删除
func (lru *SingleCache)DelByKey(key *proto.BaseKey) error { func (lru *SingleCache) DelByKey(key *proto.BaseKey) error {
if lru.lruMap == nil { if lru.lruMap == nil {
return errorx.New("lru is nil") return errorx.New("lru is nil")
} }
@ -141,3 +142,18 @@ func (lru *SingleCache)DelByKey(key *proto.BaseKey) error {
} }
return errorx.New("lru no this key") return errorx.New("lru no this key")
} }
func (lru *SingleCache) DelToClearSize() error {
if lru.lruMap == nil {
return errorx.New("lru is nil")
}
for {
if lru.nowSize > lru.clearSize {
//del自动给nowSize进行大小的改变
lru.Del()
} else {
break
}
}
return nil
}

View File

@ -3,6 +3,7 @@ package lru
import ( import (
"context" "context"
"gitee.com/timedb/wheatCache/pkg/event" "gitee.com/timedb/wheatCache/pkg/event"
"gitee.com/timedb/wheatCache/pkg/logx"
"gitee.com/timedb/wheatCache/pkg/proto" "gitee.com/timedb/wheatCache/pkg/proto"
"gitee.com/timedb/wheatCache/pkg/structure/stringx" "gitee.com/timedb/wheatCache/pkg/structure/stringx"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
@ -30,3 +31,28 @@ func TestWorker(t *testing.T) {
require.NoError(t, err) require.NoError(t, err)
require.Equal(t, res, "123") require.Equal(t, res, "123")
} }
func TestSingleCache_DelToClearSize(t *testing.T) {
ctx := context.Background()
lru := NewLRUCache()
produce := event.NewProduce(lru.GetDriver())
for i := int32(20000); i >= 1; i-- {
workEvent := event.NewEvent(OptionEventName)
workEvent.SetValue(WorkFuncEventKey, event.EventWorkFunc(func() (interface{}, error) {
v1 := stringx.NewStringSingle()
key := proto.BaseKey{
Key: string(i),
}
u := v1.Setbit(i, true)
lru.Add(&key, v1)
return u, nil
}))
workEvent.InitWaitEvent()
produce.Call(ctx, workEvent)
workEvent.StartWaitEvent(2 * time.Second)
}
time.Sleep(5 * time.Second)
logx.Info("end size is %d", lru.nowSize)
}

View File

@ -15,9 +15,7 @@ func (lru *SingleCache) lruSingleWork() interface{} {
case OptionEventName: case OptionEventName:
workFunc, ok := workEvent.GetValue(WorkFuncEventKey) workFunc, ok := workEvent.GetValue(WorkFuncEventKey)
if !ok { if !ok {
workEvent.ExecWorkAndSendResult(func() (interface{}, error) { workEvent.SetResultErr(errorx.LruNotWorkFuncEventErr())
return nil, errorx.New("the event haven't work of function")
})
continue continue
} }
@ -25,6 +23,22 @@ func (lru *SingleCache) lruSingleWork() interface{} {
workEvent.ExecWorkAndSendResult(work) workEvent.ExecWorkAndSendResult(work)
} }
case CleanEventName: case CleanEventName:
workFunc, ok := workEvent.GetValue(WorkFuncEventKey)
if !ok {
workEvent.SetResultErr(errorx.LruNotWorkFuncEventErr())
continue
}
// 对当前的io数量进行判断
ioNum := lru.GetDriver().GetLength()
if ioNum > lru.lruMaxDiverSize*1/2 {
lru.lruCleanProduce.Call(ctx, workEvent)
continue
}
if work, ok := workFunc.(event.EventWorkFunc); ok {
workEvent.ExecWorkAndSendResult(work)
}
default:
return errorx.New("no this name")
} }
} }
} }

View File

@ -3,7 +3,7 @@ package middle_msg
import "time" import "time"
var ( var (
EventNameLog = "logcontext" EventNameLog = "log-context"
) )
type LogContext struct { type LogContext struct {

14
pkg/middle-msg/lru.go Normal file
View File

@ -0,0 +1,14 @@
package middle_msg
import "time"
var LruCleanContextName = "lru-clean-context"
type LruCleanContext struct {
Keys []string
BeforeCleanSize int64
BehindCleanSize int64
StartTime time.Duration
EndTime time.Duration
}