1
0
mirror of https://github.com/astaxie/beego.git synced 2025-01-10 17:47:17 +00:00
Beego/cache/file.go

266 lines
6.6 KiB
Go
Raw Permalink Normal View History

2014-08-18 16:41:43 +08:00
// Copyright 2014 beego Author. All Rights Reserved.
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// 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
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// http://www.apache.org/licenses/LICENSE-2.0
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
2013-11-18 11:22:06 +08:00
package cache
import (
"bytes"
"crypto/md5"
"encoding/gob"
"encoding/hex"
"encoding/json"
"fmt"
"io"
"os"
"path/filepath"
2013-11-18 11:22:06 +08:00
"reflect"
"strconv"
"time"
)
func init() {
Register("file", NewFileCache())
}
2013-12-22 13:35:02 +08:00
// FileCacheItem is basic unit of file cache adapter.
// it contains data and expire time.
2013-11-18 11:22:06 +08:00
type FileCacheItem struct {
Data interface{}
Lastaccess int64
Expired int64
}
var (
2013-12-22 13:35:02 +08:00
FileCachePath string = "cache" // cache directory
2013-12-24 21:59:00 +08:00
FileCacheFileSuffix string = ".bin" // cache file suffix
FileCacheDirectoryLevel int = 2 // cache file deep level if auto generated cache files.
FileCacheEmbedExpiry int64 = 0 // cache expire time, default is no expire forever.
2013-11-18 11:22:06 +08:00
)
2013-12-22 13:35:02 +08:00
// FileCache is cache adapter for file storage.
2013-11-18 11:22:06 +08:00
type FileCache struct {
CachePath string
FileSuffix string
DirectoryLevel int
EmbedExpiry int
}
2013-12-22 15:31:49 +08:00
// Create new file cache with no config.
2013-12-22 13:35:02 +08:00
// the level and expiry need set in method StartAndGC as config string.
2013-11-18 11:22:06 +08:00
func NewFileCache() *FileCache {
2013-12-22 15:31:49 +08:00
// return &FileCache{CachePath:FileCachePath, FileSuffix:FileCacheFileSuffix}
return &FileCache{}
2013-11-18 11:22:06 +08:00
}
2013-12-22 13:35:02 +08:00
// Start and begin gc for file cache.
// the config need to be like {CachePath:"/cache","FileSuffix":".bin","DirectoryLevel":2,"EmbedExpiry":0}
2014-07-10 13:28:54 +08:00
func (fc *FileCache) StartAndGC(config string) error {
2013-11-18 11:22:06 +08:00
var cfg map[string]string
json.Unmarshal([]byte(config), &cfg)
if _, ok := cfg["CachePath"]; !ok {
cfg["CachePath"] = FileCachePath
}
if _, ok := cfg["FileSuffix"]; !ok {
cfg["FileSuffix"] = FileCacheFileSuffix
}
if _, ok := cfg["DirectoryLevel"]; !ok {
cfg["DirectoryLevel"] = strconv.Itoa(FileCacheDirectoryLevel)
}
if _, ok := cfg["EmbedExpiry"]; !ok {
cfg["EmbedExpiry"] = strconv.FormatInt(FileCacheEmbedExpiry, 10)
}
2014-07-10 13:28:54 +08:00
fc.CachePath = cfg["CachePath"]
fc.FileSuffix = cfg["FileSuffix"]
fc.DirectoryLevel, _ = strconv.Atoi(cfg["DirectoryLevel"])
fc.EmbedExpiry, _ = strconv.Atoi(cfg["EmbedExpiry"])
2013-11-18 11:22:06 +08:00
2014-07-10 13:28:54 +08:00
fc.Init()
2013-11-18 11:22:06 +08:00
return nil
}
2013-12-22 13:35:02 +08:00
// Init will make new dir for file cache if not exist.
2014-07-10 13:28:54 +08:00
func (fc *FileCache) Init() {
app := filepath.Dir(os.Args[0])
2014-07-10 13:28:54 +08:00
fc.CachePath = filepath.Join(app, fc.CachePath)
if ok, _ := exists(fc.CachePath); !ok { // todo : error handle
_ = os.MkdirAll(fc.CachePath, os.ModePerm) // todo : error handle
2013-11-18 11:22:06 +08:00
}
}
2013-12-22 13:35:02 +08:00
// get cached file name. it's md5 encoded.
2014-07-10 13:28:54 +08:00
func (fc *FileCache) getCacheFileName(key string) string {
2013-11-18 11:22:06 +08:00
m := md5.New()
io.WriteString(m, key)
keyMd5 := hex.EncodeToString(m.Sum(nil))
2014-07-10 13:28:54 +08:00
cachePath := fc.CachePath
switch fc.DirectoryLevel {
2013-11-18 11:22:06 +08:00
case 2:
cachePath = filepath.Join(cachePath, keyMd5[0:2], keyMd5[2:4])
2013-11-18 11:22:06 +08:00
case 1:
cachePath = filepath.Join(cachePath, keyMd5[0:2])
2013-11-18 11:22:06 +08:00
}
2014-07-10 13:28:54 +08:00
if ok, _ := exists(cachePath); !ok { // todo : error handle
_ = os.MkdirAll(cachePath, os.ModePerm) // todo : error handle
2013-11-18 11:22:06 +08:00
}
2014-07-10 13:28:54 +08:00
return filepath.Join(cachePath, fmt.Sprintf("%s%s", keyMd5, fc.FileSuffix))
2013-11-18 11:22:06 +08:00
}
2013-12-22 13:35:02 +08:00
// Get value from file cache.
// if non-exist or expired, return empty string.
2014-07-10 13:28:54 +08:00
func (fc *FileCache) Get(key string) interface{} {
fileData, err := File_get_contents(fc.getCacheFileName(key))
2013-11-18 11:22:06 +08:00
if err != nil {
return ""
}
var to FileCacheItem
2014-07-10 13:28:54 +08:00
Gob_decode(fileData, &to)
2013-11-18 11:22:06 +08:00
if to.Expired < time.Now().Unix() {
return ""
}
return to.Data
}
2013-12-22 13:35:02 +08:00
// Put value into file cache.
2013-12-22 15:31:49 +08:00
// timeout means how long to keep this file, unit of ms.
// if timeout equals FileCacheEmbedExpiry(default is 0), cache this item forever.
2014-07-10 13:28:54 +08:00
func (fc *FileCache) Put(key string, val interface{}, timeout int64) error {
gob.Register(val)
2014-07-10 13:38:05 +08:00
item := FileCacheItem{Data: val}
2013-11-18 11:22:06 +08:00
if timeout == FileCacheEmbedExpiry {
2014-07-10 13:38:05 +08:00
item.Expired = time.Now().Unix() + (86400 * 365 * 10) // ten years
2013-11-18 11:22:06 +08:00
} else {
2014-07-10 13:38:05 +08:00
item.Expired = time.Now().Unix() + timeout
2013-11-18 11:22:06 +08:00
}
item.Lastaccess = time.Now().Unix()
data, err := Gob_encode(item)
if err != nil {
return err
}
2014-07-10 13:28:54 +08:00
return File_put_contents(fc.getCacheFileName(key), data)
2013-11-18 11:22:06 +08:00
}
2013-12-22 13:35:02 +08:00
// Delete file cache value.
2014-07-10 13:28:54 +08:00
func (fc *FileCache) Delete(key string) error {
filename := fc.getCacheFileName(key)
2013-11-18 11:22:06 +08:00
if ok, _ := exists(filename); ok {
return os.Remove(filename)
}
return nil
}
2013-12-22 13:35:02 +08:00
// Increase cached int value.
2014-07-10 13:28:54 +08:00
// fc value is saving forever unless Delete.
func (fc *FileCache) Incr(key string) error {
data := fc.Get(key)
2013-11-18 11:22:06 +08:00
var incr int
if reflect.TypeOf(data).Name() != "int" {
incr = 0
} else {
2014-07-10 13:38:05 +08:00
incr = data.(int) + 1
2013-11-18 11:22:06 +08:00
}
2014-07-10 13:28:54 +08:00
fc.Put(key, incr, FileCacheEmbedExpiry)
2013-11-18 11:22:06 +08:00
return nil
}
2013-12-22 13:35:02 +08:00
// Decrease cached int value.
2014-07-10 13:28:54 +08:00
func (fc *FileCache) Decr(key string) error {
data := fc.Get(key)
2013-11-18 11:22:06 +08:00
var decr int
if reflect.TypeOf(data).Name() != "int" || data.(int)-1 <= 0 {
decr = 0
} else {
2014-07-10 13:38:05 +08:00
decr = data.(int) - 1
2013-11-18 11:22:06 +08:00
}
2014-07-10 13:28:54 +08:00
fc.Put(key, decr, FileCacheEmbedExpiry)
2013-11-18 11:22:06 +08:00
return nil
}
2013-12-22 13:35:02 +08:00
// Check value is exist.
2014-07-10 13:28:54 +08:00
func (fc *FileCache) IsExist(key string) bool {
ret, _ := exists(fc.getCacheFileName(key))
2013-11-18 11:22:06 +08:00
return ret
}
2013-12-22 13:35:02 +08:00
// Clean cached files.
// not implemented.
2014-07-10 13:28:54 +08:00
func (fc *FileCache) ClearAll() error {
2013-11-18 11:22:06 +08:00
return nil
}
2013-12-22 13:35:02 +08:00
// check file exist.
2013-11-18 11:22:06 +08:00
func exists(path string) (bool, error) {
_, err := os.Stat(path)
if err == nil {
return true, nil
}
if os.IsNotExist(err) {
return false, nil
}
return false, err
}
2013-12-22 13:35:02 +08:00
// Get bytes to file.
// if non-exist, create this file.
2014-07-10 13:28:54 +08:00
func File_get_contents(filename string) (data []byte, e error) {
f, e := os.OpenFile(filename, os.O_RDWR|os.O_CREATE, os.ModePerm)
if e != nil {
return
2013-11-18 11:22:06 +08:00
}
defer f.Close()
2014-07-10 13:28:54 +08:00
stat, e := f.Stat()
if e != nil {
return
2013-11-18 11:22:06 +08:00
}
2014-07-10 13:28:54 +08:00
data = make([]byte, stat.Size())
result, e := f.Read(data)
if e != nil || int64(result) != stat.Size() {
return nil, e
2013-11-18 11:22:06 +08:00
}
2014-07-10 13:28:54 +08:00
return
2013-11-18 11:22:06 +08:00
}
2013-12-22 13:35:02 +08:00
// Put bytes to file.
// if non-exist, create this file.
2013-11-18 11:22:06 +08:00
func File_put_contents(filename string, content []byte) error {
fp, err := os.OpenFile(filename, os.O_RDWR|os.O_CREATE, os.ModePerm)
if err != nil {
return err
}
defer fp.Close()
_, err = fp.Write(content)
return err
}
2013-12-22 13:35:02 +08:00
// Gob encodes file cache item.
2013-11-18 11:22:06 +08:00
func Gob_encode(data interface{}) ([]byte, error) {
buf := bytes.NewBuffer(nil)
enc := gob.NewEncoder(buf)
err := enc.Encode(data)
if err != nil {
return nil, err
}
return buf.Bytes(), err
}
2013-12-22 13:35:02 +08:00
// Gob decodes file cache item.
2014-01-01 20:50:06 +08:00
func Gob_decode(data []byte, to *FileCacheItem) error {
2013-11-18 11:22:06 +08:00
buf := bytes.NewBuffer(data)
dec := gob.NewDecoder(buf)
return dec.Decode(&to)
}