Feat: cache in-memory store

pull/247/head
HFO4 5 years ago
parent 5d50e7ed1e
commit 7375cc01f1

@ -5,6 +5,7 @@ go 1.12
require ( require (
github.com/DATA-DOG/go-sqlmock v1.3.3 github.com/DATA-DOG/go-sqlmock v1.3.3
github.com/fatih/color v1.7.0 github.com/fatih/color v1.7.0
github.com/garyburd/redigo v1.6.0
github.com/gin-contrib/cors v1.3.0 github.com/gin-contrib/cors v1.3.0
github.com/gin-contrib/pprof v1.2.1 github.com/gin-contrib/pprof v1.2.1
github.com/gin-contrib/sessions v0.0.1 github.com/gin-contrib/sessions v0.0.1

@ -1,16 +1,17 @@
package model package model
import ( import (
"github.com/gin-gonic/gin" "github.com/jinzhu/gorm"
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
"testing" "testing"
) )
func TestMigration(t *testing.T) { func TestMigration(t *testing.T) {
asserts := assert.New(t) asserts := assert.New(t)
gin.SetMode(gin.TestMode) DB, _ = gorm.Open("sqlite3", ":memory:")
asserts.NotPanics(func() { asserts.NotPanics(func() {
migration() migration()
}) })
DB = mockDB
} }

@ -2,11 +2,11 @@ package model
import ( import (
"encoding/json" "encoding/json"
"github.com/HFO4/cloudreve/pkg/cache"
"github.com/HFO4/cloudreve/pkg/util" "github.com/HFO4/cloudreve/pkg/util"
"github.com/jinzhu/gorm" "github.com/jinzhu/gorm"
"path/filepath" "path/filepath"
"strconv" "strconv"
"sync"
"time" "time"
) )
@ -42,28 +42,20 @@ type PolicyOption struct {
RangeTransferEnabled bool `json:"range_transfer_enabled"` RangeTransferEnabled bool `json:"range_transfer_enabled"`
} }
// 存储策略缓存,部分情况下需要频繁查询存储策略
var policyCache = make(map[uint]Policy)
var rw sync.RWMutex
// GetPolicyByID 用ID获取存储策略 // GetPolicyByID 用ID获取存储策略
func GetPolicyByID(ID interface{}) (Policy, error) { func GetPolicyByID(ID interface{}) (Policy, error) {
// 尝试读取缓存 // 尝试读取缓存
rw.RLock() cacheKey := "policy_" + strconv.Itoa(int(ID.(uint)))
if policy, ok := policyCache[ID.(uint)]; ok { if policy, ok := cache.Store.Get(cacheKey); ok {
rw.RUnlock() return policy.(Policy), nil
return policy, nil
} }
rw.RUnlock()
var policy Policy var policy Policy
result := DB.First(&policy, ID) result := DB.First(&policy, ID)
// 写入缓存 // 写入缓存
if result.Error == nil { if result.Error == nil {
rw.Lock() _ = cache.Store.Set(cacheKey, policy)
policyCache[policy.ID] = policy
rw.Unlock()
} }
return policy, result.Error return policy, result.Error

@ -12,20 +12,35 @@ import (
func TestGetPolicyByID(t *testing.T) { func TestGetPolicyByID(t *testing.T) {
asserts := assert.New(t) asserts := assert.New(t)
// 缓存未命中
{
rows := sqlmock.NewRows([]string{"name", "type", "options"}). rows := sqlmock.NewRows([]string{"name", "type", "options"}).
AddRow("默认存储策略", "local", "{\"op_name\":\"123\"}") AddRow("默认存储策略", "local", "{\"op_name\":\"123\"}")
mock.ExpectQuery("^SELECT \\* FROM `(.+)` WHERE `(.+)`\\.`deleted_at` IS NULL AND \\(\\(`policies`.`id` = 1\\)\\)(.+)$").WillReturnRows(rows) mock.ExpectQuery("^SELECT(.+)").WillReturnRows(rows)
policy, err := GetPolicyByID(uint(1)) policy, err := GetPolicyByID(uint(22))
asserts.NoError(err) asserts.NoError(err)
asserts.NoError(mock.ExpectationsWereMet())
asserts.Equal("默认存储策略", policy.Name) asserts.Equal("默认存储策略", policy.Name)
asserts.Equal("123", policy.OptionsSerialized.OPName) asserts.Equal("123", policy.OptionsSerialized.OPName)
rows = sqlmock.NewRows([]string{"name", "type", "options"}) rows = sqlmock.NewRows([]string{"name", "type", "options"})
mock.ExpectQuery("^SELECT \\* FROM `(.+)` WHERE `(.+)`\\.`deleted_at` IS NULL AND \\(\\(`policies`.`id` = 1\\)\\)(.+)$").WillReturnRows(rows) mock.ExpectQuery("^SELECT(.+)").WillReturnRows(rows)
policy, err = GetPolicyByID(uint(1)) policy, err = GetPolicyByID(uint(23))
asserts.NoError(mock.ExpectationsWereMet())
asserts.Error(err) asserts.Error(err)
} }
// 命中
{
policy, err := GetPolicyByID(uint(22))
asserts.NoError(err)
asserts.Equal("默认存储策略", policy.Name)
asserts.Equal("123", policy.OptionsSerialized.OPName)
}
}
func TestPolicy_BeforeSave(t *testing.T) { func TestPolicy_BeforeSave(t *testing.T) {
asserts := assert.New(t) asserts := assert.New(t)

@ -9,6 +9,7 @@ import (
) )
var mock sqlmock.Sqlmock var mock sqlmock.Sqlmock
var mockDB *gorm.DB
// TestMain 初始化数据库Mock // TestMain 初始化数据库Mock
func TestMain(m *testing.M) { func TestMain(m *testing.M) {
@ -19,6 +20,7 @@ func TestMain(m *testing.M) {
panic("An error was not expected when opening a stub database connection") panic("An error was not expected when opening a stub database connection")
} }
DB, _ = gorm.Open("mysql", db) DB, _ = gorm.Open("mysql", db)
mockDB = DB
defer db.Close() defer db.Close()
m.Run() m.Run()
} }

@ -0,0 +1,10 @@
package cache
// Store 缓存存储器
var Store Driver = NewMemoStore()
// Driver 键值缓存存储容器
type Driver interface {
Set(key string, value interface{}) error
Get(key string) (interface{}, bool)
}

26
pkg/cache/memo.go vendored

@ -0,0 +1,26 @@
package cache
import "sync"
// MemoStore 内存存储驱动
type MemoStore struct {
Store *sync.Map
}
// NewMemoStore 新建内存存储
func NewMemoStore() *MemoStore {
return &MemoStore{
Store: &sync.Map{},
}
}
// Set 存储值
func (store *MemoStore) Set(key string, value interface{}) error {
store.Store.Store(key, value)
return nil
}
// Get 取值
func (store *MemoStore) Get(key string) (interface{}, bool) {
return store.Store.Load(key)
}

@ -0,0 +1,61 @@
package cache
import (
"github.com/stretchr/testify/assert"
"testing"
)
func TestNewMemoStore(t *testing.T) {
asserts := assert.New(t)
store := NewMemoStore()
asserts.NotNil(store)
asserts.NotNil(store.Store)
}
func TestMemoStore_Set(t *testing.T) {
asserts := assert.New(t)
store := NewMemoStore()
err := store.Set("KEY", "vAL")
asserts.NoError(err)
val, ok := store.Store.Load("KEY")
asserts.True(ok)
asserts.Equal("vAL", val)
}
func TestMemoStore_Get(t *testing.T) {
asserts := assert.New(t)
store := NewMemoStore()
// 正常情况
{
_ = store.Set("string", "string_val")
val, ok := store.Get("string")
asserts.Equal("string_val", val)
asserts.True(ok)
}
// Key不存在
{
val, ok := store.Get("something")
asserts.Equal(nil, val)
asserts.False(ok)
}
// 存储struct
{
type testStruct struct {
key int
}
test := testStruct{key: 233}
_ = store.Set("struct", test)
val, ok := store.Get("struct")
asserts.True(ok)
res, ok := val.(testStruct)
asserts.True(ok)
asserts.Equal(test, res)
}
}
Loading…
Cancel
Save