2020-04-11 19:45:24 +00:00
|
|
|
package chunk_cache
|
2020-03-28 20:43:31 +00:00
|
|
|
|
|
|
|
import (
|
2020-04-12 04:12:41 +00:00
|
|
|
"fmt"
|
|
|
|
"path"
|
|
|
|
"sort"
|
|
|
|
"sync"
|
2020-03-28 20:43:31 +00:00
|
|
|
|
2020-04-12 04:12:41 +00:00
|
|
|
"github.com/chrislusf/seaweedfs/weed/glog"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/storage"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/storage/needle"
|
2020-03-28 20:43:31 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// a global cache for recently accessed file chunks
|
|
|
|
type ChunkCache struct {
|
2020-04-12 04:12:41 +00:00
|
|
|
memCache *ChunkCacheInMemory
|
|
|
|
diskCaches []*ChunkCacheVolume
|
|
|
|
sync.RWMutex
|
2020-03-28 20:43:31 +00:00
|
|
|
}
|
|
|
|
|
2020-04-12 04:12:41 +00:00
|
|
|
func NewChunkCache(maxEntries int64, dir string, diskSizeMB int64, segmentCount int) *ChunkCache {
|
|
|
|
c := &ChunkCache{
|
|
|
|
memCache: NewChunkCacheInMemory(maxEntries),
|
2020-03-28 21:07:16 +00:00
|
|
|
}
|
2020-04-12 04:12:41 +00:00
|
|
|
|
|
|
|
volumeCount, volumeSize := int(diskSizeMB/30000), int64(30000)
|
|
|
|
if volumeCount < segmentCount {
|
|
|
|
volumeCount, volumeSize = segmentCount, diskSizeMB/int64(segmentCount)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < volumeCount; i++ {
|
|
|
|
fileName := path.Join(dir, fmt.Sprintf("cache_%d", i))
|
|
|
|
diskCache, err := LoadOrCreateChunkCacheVolume(fileName, volumeSize*1024*1024)
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("failed to add cache %s : %v", fileName, err)
|
|
|
|
} else {
|
|
|
|
c.diskCaches = append(c.diskCaches, diskCache)
|
|
|
|
}
|
2020-03-28 20:43:31 +00:00
|
|
|
}
|
2020-04-12 04:12:41 +00:00
|
|
|
|
|
|
|
// keep newest cache to the front
|
|
|
|
sort.Slice(c.diskCaches, func(i, j int) bool {
|
|
|
|
return c.diskCaches[i].lastModTime.After(c.diskCaches[j].lastModTime)
|
|
|
|
})
|
|
|
|
|
|
|
|
return c
|
2020-03-28 20:43:31 +00:00
|
|
|
}
|
|
|
|
|
2020-04-12 04:12:41 +00:00
|
|
|
func (c *ChunkCache) GetChunk(fileId string) (data []byte) {
|
2020-04-12 08:00:12 +00:00
|
|
|
if c == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-04-12 04:12:41 +00:00
|
|
|
c.RLock()
|
|
|
|
defer c.RUnlock()
|
|
|
|
|
2020-04-12 08:06:50 +00:00
|
|
|
return c.doGetChunk(fileId)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *ChunkCache) doGetChunk(fileId string) (data []byte) {
|
2020-04-12 04:12:41 +00:00
|
|
|
if data = c.memCache.GetChunk(fileId); data != nil {
|
|
|
|
return data
|
|
|
|
}
|
|
|
|
|
|
|
|
fid, err := needle.ParseFileIdFromString(fileId)
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("failed to parse file id %s", fileId)
|
2020-03-28 20:43:31 +00:00
|
|
|
return nil
|
|
|
|
}
|
2020-04-12 04:12:41 +00:00
|
|
|
for _, diskCache := range c.diskCaches {
|
|
|
|
data, err = diskCache.GetNeedle(fid.Key)
|
|
|
|
if err == storage.ErrorNotFound {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("failed to read cache file %s id %s", diskCache.fileName, fileId)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if len(data) != 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2020-03-28 20:43:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *ChunkCache) SetChunk(fileId string, data []byte) {
|
2020-04-12 08:00:12 +00:00
|
|
|
if c == nil {
|
|
|
|
return
|
|
|
|
}
|
2020-04-12 04:12:41 +00:00
|
|
|
c.Lock()
|
|
|
|
defer c.Unlock()
|
|
|
|
|
2020-04-12 10:34:36 +00:00
|
|
|
c.doSetChunk(fileId, data)
|
2020-04-12 08:06:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *ChunkCache) doSetChunk(fileId string, data []byte) {
|
|
|
|
|
2020-04-12 04:12:41 +00:00
|
|
|
c.memCache.SetChunk(fileId, data)
|
|
|
|
|
|
|
|
if len(c.diskCaches) == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.diskCaches[0].fileSize+int64(len(data)) > c.diskCaches[0].sizeLimit {
|
|
|
|
t, resetErr := c.diskCaches[len(c.diskCaches)-1].Reset()
|
|
|
|
if resetErr != nil {
|
|
|
|
glog.Errorf("failed to reset cache file %s", c.diskCaches[len(c.diskCaches)-1].fileName)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for i := len(c.diskCaches) - 1; i > 0; i-- {
|
|
|
|
c.diskCaches[i] = c.diskCaches[i-1]
|
|
|
|
}
|
|
|
|
c.diskCaches[0] = t
|
|
|
|
}
|
|
|
|
|
|
|
|
fid, err := needle.ParseFileIdFromString(fileId)
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("failed to parse file id %s", fileId)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.diskCaches[0].WriteNeedle(fid.Key, data)
|
|
|
|
|
2020-03-28 20:43:31 +00:00
|
|
|
}
|
2020-04-12 04:12:41 +00:00
|
|
|
|
|
|
|
func (c *ChunkCache) Shutdown() {
|
2020-04-12 08:00:12 +00:00
|
|
|
if c == nil {
|
|
|
|
return
|
|
|
|
}
|
2020-04-12 04:12:41 +00:00
|
|
|
c.Lock()
|
|
|
|
defer c.Unlock()
|
|
|
|
for _, diskCache := range c.diskCaches {
|
|
|
|
diskCache.Shutdown()
|
|
|
|
}
|
|
|
|
}
|