simple.go 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273
  1. package gcache
  2. import "time"
  3. // SimpleCache has no clear priority for evict cache. It depends on key-value map order.
  4. type SimpleCache struct {
  5. baseCache
  6. items map[interface{}]*simpleItem
  7. }
  8. func newSimpleCache(cb *CacheBuilder) *SimpleCache {
  9. c := &SimpleCache{}
  10. buildCache(&c.baseCache, cb)
  11. c.init()
  12. c.loadGroup.cache = c
  13. return c
  14. }
  15. func (c *SimpleCache) init() {
  16. if c.size <= 0 {
  17. c.items = make(map[interface{}]*simpleItem)
  18. } else {
  19. c.items = make(map[interface{}]*simpleItem, c.size)
  20. }
  21. }
  22. // Set a new key-value pair
  23. func (c *SimpleCache) Set(key, value interface{}) error {
  24. c.mu.Lock()
  25. defer c.mu.Unlock()
  26. _, err := c.set(key, value)
  27. return err
  28. }
  29. // Set a new key-value pair with an expiration time
  30. func (c *SimpleCache) SetWithExpire(key, value interface{}, expiration time.Duration) error {
  31. c.mu.Lock()
  32. defer c.mu.Unlock()
  33. item, err := c.set(key, value)
  34. if err != nil {
  35. return err
  36. }
  37. t := c.clock.Now().Add(expiration)
  38. item.(*simpleItem).expiration = &t
  39. return nil
  40. }
  41. func (c *SimpleCache) set(key, value interface{}) (interface{}, error) {
  42. var err error
  43. if c.serializeFunc != nil {
  44. value, err = c.serializeFunc(key, value)
  45. if err != nil {
  46. return nil, err
  47. }
  48. }
  49. // Check for existing item
  50. item, ok := c.items[key]
  51. if ok {
  52. item.value = value
  53. } else {
  54. // Verify size not exceeded
  55. if (len(c.items) >= c.size) && c.size > 0 {
  56. c.evict(1)
  57. }
  58. item = &simpleItem{
  59. clock: c.clock,
  60. value: value,
  61. }
  62. c.items[key] = item
  63. }
  64. if c.expiration != nil {
  65. t := c.clock.Now().Add(*c.expiration)
  66. item.expiration = &t
  67. }
  68. if c.addedFunc != nil {
  69. c.addedFunc(key, value)
  70. }
  71. return item, nil
  72. }
  73. // Get a value from cache pool using key if it exists.
  74. // If it dose not exists key and has LoaderFunc,
  75. // generate a value using `LoaderFunc` method returns value.
  76. func (c *SimpleCache) Get(key interface{}) (interface{}, error) {
  77. v, err := c.get(key, false)
  78. if err == KeyNotFoundError {
  79. return c.getWithLoader(key, true)
  80. }
  81. return v, err
  82. }
  83. // Get a value from cache pool using key if it exists.
  84. // If it dose not exists key, returns KeyNotFoundError.
  85. // And send a request which refresh value for specified key if cache object has LoaderFunc.
  86. func (c *SimpleCache) GetIFPresent(key interface{}) (interface{}, error) {
  87. v, err := c.get(key, false)
  88. if err == KeyNotFoundError {
  89. return c.getWithLoader(key, false)
  90. }
  91. return v, nil
  92. }
  93. func (c *SimpleCache) get(key interface{}, onLoad bool) (interface{}, error) {
  94. v, err := c.getValue(key, onLoad)
  95. if err != nil {
  96. return nil, err
  97. }
  98. if c.deserializeFunc != nil {
  99. return c.deserializeFunc(key, v)
  100. }
  101. return v, nil
  102. }
  103. func (c *SimpleCache) getValue(key interface{}, onLoad bool) (interface{}, error) {
  104. c.mu.Lock()
  105. item, ok := c.items[key]
  106. if ok {
  107. if !item.IsExpired(nil) {
  108. v := item.value
  109. c.mu.Unlock()
  110. if !onLoad {
  111. c.stats.IncrHitCount()
  112. }
  113. return v, nil
  114. }
  115. c.remove(key)
  116. }
  117. c.mu.Unlock()
  118. if !onLoad {
  119. c.stats.IncrMissCount()
  120. }
  121. return nil, KeyNotFoundError
  122. }
  123. func (c *SimpleCache) getWithLoader(key interface{}, isWait bool) (interface{}, error) {
  124. if c.loaderExpireFunc == nil {
  125. return nil, KeyNotFoundError
  126. }
  127. value, _, err := c.load(key, func(v interface{}, expiration *time.Duration, e error) (interface{}, error) {
  128. if e != nil {
  129. return nil, e
  130. }
  131. c.mu.Lock()
  132. defer c.mu.Unlock()
  133. item, err := c.set(key, v)
  134. if err != nil {
  135. return nil, err
  136. }
  137. if expiration != nil {
  138. t := c.clock.Now().Add(*expiration)
  139. item.(*simpleItem).expiration = &t
  140. }
  141. return v, nil
  142. }, isWait)
  143. if err != nil {
  144. return nil, err
  145. }
  146. return value, nil
  147. }
  148. func (c *SimpleCache) evict(count int) {
  149. now := c.clock.Now()
  150. current := 0
  151. for key, item := range c.items {
  152. if current >= count {
  153. return
  154. }
  155. if item.expiration == nil || now.After(*item.expiration) {
  156. defer c.remove(key)
  157. current++
  158. }
  159. }
  160. }
  161. // Removes the provided key from the cache.
  162. func (c *SimpleCache) Remove(key interface{}) bool {
  163. c.mu.Lock()
  164. defer c.mu.Unlock()
  165. return c.remove(key)
  166. }
  167. func (c *SimpleCache) remove(key interface{}) bool {
  168. item, ok := c.items[key]
  169. if ok {
  170. delete(c.items, key)
  171. if c.evictedFunc != nil {
  172. c.evictedFunc(key, item.value)
  173. }
  174. return true
  175. }
  176. return false
  177. }
  178. // Returns a slice of the keys in the cache.
  179. func (c *SimpleCache) keys() []interface{} {
  180. c.mu.RLock()
  181. defer c.mu.RUnlock()
  182. keys := make([]interface{}, len(c.items))
  183. var i = 0
  184. for k := range c.items {
  185. keys[i] = k
  186. i++
  187. }
  188. return keys
  189. }
  190. // Returns a slice of the keys in the cache.
  191. func (c *SimpleCache) Keys() []interface{} {
  192. keys := []interface{}{}
  193. for _, k := range c.keys() {
  194. _, err := c.GetIFPresent(k)
  195. if err == nil {
  196. keys = append(keys, k)
  197. }
  198. }
  199. return keys
  200. }
  201. // Returns all key-value pairs in the cache.
  202. func (c *SimpleCache) GetALL() map[interface{}]interface{} {
  203. m := make(map[interface{}]interface{})
  204. for _, k := range c.keys() {
  205. v, err := c.GetIFPresent(k)
  206. if err == nil {
  207. m[k] = v
  208. }
  209. }
  210. return m
  211. }
  212. // Returns the number of items in the cache.
  213. func (c *SimpleCache) Len() int {
  214. return len(c.GetALL())
  215. }
  216. // Completely clear the cache
  217. func (c *SimpleCache) Purge() {
  218. c.mu.Lock()
  219. defer c.mu.Unlock()
  220. if c.purgeVisitorFunc != nil {
  221. for key, item := range c.items {
  222. c.purgeVisitorFunc(key, item.value)
  223. }
  224. }
  225. c.init()
  226. }
  227. type simpleItem struct {
  228. clock Clock
  229. value interface{}
  230. expiration *time.Time
  231. }
  232. // returns boolean value whether this item is expired or not.
  233. func (si *simpleItem) IsExpired(now *time.Time) bool {
  234. if si.expiration == nil {
  235. return false
  236. }
  237. if now == nil {
  238. t := si.clock.Now()
  239. now = &t
  240. }
  241. return si.expiration.Before(*now)
  242. }