1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118
|
// Copyright (c) 2012-2016 The Revel Framework Authors, All rights reserved.
// Revel Framework source code and usage is governed by a MIT style
// license that can be found in the LICENSE file.
package cache
import (
"errors"
"time"
"github.com/bradfitz/gomemcache/memcache"
"github.com/revel/revel/logger"
)
// MemcachedCache wraps the Memcached client to meet the Cache interface.
type MemcachedCache struct {
*memcache.Client
defaultExpiration time.Duration
}
func NewMemcachedCache(hostList []string, defaultExpiration time.Duration) MemcachedCache {
return MemcachedCache{memcache.New(hostList...), defaultExpiration}
}
func (c MemcachedCache) Set(key string, value interface{}, expires time.Duration) error {
return c.invoke((*memcache.Client).Set, key, value, expires)
}
func (c MemcachedCache) Add(key string, value interface{}, expires time.Duration) error {
return c.invoke((*memcache.Client).Add, key, value, expires)
}
func (c MemcachedCache) Replace(key string, value interface{}, expires time.Duration) error {
return c.invoke((*memcache.Client).Replace, key, value, expires)
}
func (c MemcachedCache) Get(key string, ptrValue interface{}) error {
item, err := c.Client.Get(key)
if err != nil {
return convertMemcacheError(err)
}
return Deserialize(item.Value, ptrValue)
}
func (c MemcachedCache) GetMulti(keys ...string) (Getter, error) {
items, err := c.Client.GetMulti(keys)
if err != nil {
return nil, convertMemcacheError(err)
}
return ItemMapGetter(items), nil
}
func (c MemcachedCache) Delete(key string) error {
return convertMemcacheError(c.Client.Delete(key))
}
func (c MemcachedCache) Increment(key string, delta uint64) (newValue uint64, err error) {
newValue, err = c.Client.Increment(key, delta)
return newValue, convertMemcacheError(err)
}
func (c MemcachedCache) Decrement(key string, delta uint64) (newValue uint64, err error) {
newValue, err = c.Client.Decrement(key, delta)
return newValue, convertMemcacheError(err)
}
func (c MemcachedCache) Flush() error {
err := errors.New("Flush: can not flush memcached")
cacheLog.Error(err.Error())
return err
}
func (c MemcachedCache) invoke(f func(*memcache.Client, *memcache.Item) error,
key string, value interface{}, expires time.Duration) error {
switch expires {
case DefaultExpiryTime:
expires = c.defaultExpiration
case ForEverNeverExpiry:
expires = time.Duration(0)
}
b, err := Serialize(value)
if err != nil {
return err
}
return convertMemcacheError(f(c.Client, &memcache.Item{
Key: key,
Value: b,
Expiration: int32(expires / time.Second),
}))
}
// ItemMapGetter implements a Getter on top of the returned item map.
type ItemMapGetter map[string]*memcache.Item
func (g ItemMapGetter) Get(key string, ptrValue interface{}) error {
item, ok := g[key]
if !ok {
return ErrCacheMiss
}
return Deserialize(item.Value, ptrValue)
}
func convertMemcacheError(err error) error {
switch err {
case nil:
return nil
case memcache.ErrCacheMiss:
return ErrCacheMiss
case memcache.ErrNotStored:
return ErrNotStored
}
cacheLog.Error("convertMemcacheError:", "error", err, "trace", logger.NewCallStack())
return err
}
|