cachemap.go 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. package cache
  2. import (
  3. "context"
  4. "sync"
  5. "time"
  6. )
  7. const (
  8. LiveInfinite = time.Duration(-1)
  9. )
  10. type CallCreate func(key any) (value any, created bool)
  11. type CallCreateNew func(key any) (rKey, value any, created bool)
  12. type CallCheck func(key, value any, exists bool) (rKey, rValue any, created bool)
  13. type item struct {
  14. value any
  15. expire int64
  16. }
  17. func NewCacheMap(ctx context.Context, maxSize int, live time.Duration) *CacheMap {
  18. cctx, cancel := context.WithCancel(ctx)
  19. res := &CacheMap{
  20. ctx: cctx,
  21. cancel: cancel,
  22. items: make(map[any]*item),
  23. mi: new(sync.RWMutex),
  24. live: live,
  25. maxSize: maxSize,
  26. mc: new(sync.RWMutex),
  27. }
  28. // todo - clean worker
  29. return res
  30. }
  31. type CacheMap struct {
  32. finished bool
  33. ctx context.Context
  34. cancel context.CancelFunc
  35. items map[any]*item
  36. mi *sync.RWMutex
  37. live time.Duration
  38. maxSize int
  39. cleanChans []chan map[any]any
  40. mc *sync.RWMutex
  41. }
  42. func (s *CacheMap) IsFinished() bool {
  43. return s.finished
  44. }
  45. func (s *CacheMap) close() {
  46. s.finished = true
  47. s.mc.Lock()
  48. for _, ch := range s.cleanChans {
  49. close(ch)
  50. }
  51. s.cleanChans = nil
  52. s.mc.Unlock()
  53. }
  54. func (s *CacheMap) eventClean(m map[any]any) {
  55. s.mc.RLock()
  56. for _, ch := range s.cleanChans {
  57. select {
  58. case ch <- m:
  59. default:
  60. }
  61. }
  62. s.mc.RUnlock()
  63. }
  64. func (s *CacheMap) EventCleaner() <-chan map[any]any {
  65. s.mc.Lock()
  66. ech := make(chan map[any]any, 1)
  67. s.cleanChans = append(s.cleanChans, ech)
  68. s.mc.Unlock()
  69. return ech
  70. }
  71. func (s *CacheMap) cleanWorker() {
  72. t := time.NewTicker(s.live)
  73. loop:
  74. for {
  75. select {
  76. case <-t.C:
  77. // clean items
  78. nowTS := time.Now().Unix()
  79. m := make(map[any]any)
  80. s.mi.Lock()
  81. for key, item := range s.items {
  82. if item.expire < nowTS {
  83. delete(s.items, key)
  84. m[key] = item.value
  85. }
  86. }
  87. s.mi.Unlock()
  88. if len(m) > 0 {
  89. s.eventClean(m)
  90. }
  91. case <-s.ctx.Done():
  92. break loop
  93. }
  94. }
  95. }
  96. func (s *CacheMap) delete(key any) {
  97. delete(s.items, key)
  98. }
  99. // Delete removes cached object
  100. func (s *CacheMap) Delete(key any) {
  101. s.mi.Lock()
  102. s.delete(key)
  103. s.mi.Unlock()
  104. }
  105. func (s *CacheMap) set(key, value any) {
  106. s.items[key] = &item{
  107. value: value,
  108. expire: time.Now().Add(s.live).UnixNano(),
  109. }
  110. }
  111. func (s *CacheMap) keysOlder(count int) []any {
  112. if count > len(s.items) {
  113. count = len(s.items)
  114. }
  115. last, llast := time.Now().Unix(), int64(-1)
  116. res := make([]any, 0, count)
  117. for len(res) < count {
  118. var rKey any
  119. ltmp := last
  120. for key, val := range s.items {
  121. if val.expire < ltmp && val.expire > llast {
  122. rKey = key
  123. ltmp = val.expire
  124. }
  125. }
  126. res = append(res, rKey)
  127. llast = ltmp
  128. }
  129. return res
  130. }
  131. func (s *CacheMap) Set(key, value any) {
  132. s.mi.Lock()
  133. if s.maxSize > 0 && len(s.items) == s.maxSize {
  134. // remove first
  135. for key, val := range s.items {
  136. delete(s.items, key)
  137. s.eventClean(map[any]any{
  138. key: val.value,
  139. })
  140. break
  141. }
  142. }
  143. s.set(key, value)
  144. s.mi.Unlock()
  145. }
  146. func (s *CacheMap) SetMulti(m map[any]any) {
  147. s.mi.Lock()
  148. /*
  149. if s.maxSize > 0 && len(s.items)+len(m) > s.maxSize {
  150. // remove first
  151. for key, val := range s.items {
  152. delete(s.items, key)
  153. s.eventClean(map[any]any{
  154. key: val.value,
  155. })
  156. break
  157. }
  158. }
  159. */
  160. for key, val := range m {
  161. s.set(key, val)
  162. }
  163. s.mi.Unlock()
  164. }
  165. func (s *CacheMap) DeleteMulti(keys []any) {
  166. s.mi.Lock()
  167. for _, key := range keys {
  168. delete(s.items, key)
  169. }
  170. s.mi.Unlock()
  171. }
  172. func (s *CacheMap) get(key any) (res any, check bool) {
  173. var it *item
  174. if it, check = s.items[key]; check {
  175. res = it.value
  176. }
  177. return
  178. }
  179. func (s *CacheMap) Get(key any) (res any, check bool) {
  180. s.mi.RLock()
  181. res, check = s.get(key)
  182. s.mi.RUnlock()
  183. return
  184. }
  185. func (s *CacheMap) GetCreate(key any, mCreate CallCreate) (val any, check bool) {
  186. if val, check = s.Get(key); !check {
  187. s.mi.Lock()
  188. defer s.mi.Unlock()
  189. if val, check = s.get(key); check {
  190. return
  191. }
  192. if val, check = mCreate(key); check {
  193. s.set(key, val)
  194. }
  195. }
  196. return
  197. }
  198. func (s *CacheMap) GetCreateNew(key any, mCreateNew CallCreateNew) (rKey, val any, check bool) {
  199. rKey = key
  200. if val, check = s.Get(key); !check {
  201. s.mi.Lock()
  202. defer s.mi.Unlock()
  203. if val, check = s.get(key); check {
  204. return
  205. }
  206. if rKey, val, check = mCreateNew(key); check {
  207. s.set(rKey, val)
  208. }
  209. }
  210. return
  211. }
  212. func (s *CacheMap) GetCheck(key any, mCheck CallCheck) (res any, check bool) {
  213. s.mi.Lock()
  214. res, check = s.get(key)
  215. if rKey, rVal, rCheck := mCheck(key, res, check); rCheck {
  216. s.set(rKey, rVal)
  217. key, res, check = rKey, rVal, true
  218. }
  219. s.mi.Unlock()
  220. return
  221. }
  222. // Each implements a map bypass for each key using the callback function. If the callback function returns false, then the cycle stops
  223. func (s *CacheMap) Each(callback func(any, any) bool) {
  224. s.mi.RLock()
  225. defer s.mi.RUnlock()
  226. for key, val := range s.items {
  227. if !callback(key, val.value) {
  228. return
  229. }
  230. }
  231. }
  232. func (s CacheMap) Len() (res int) {
  233. s.mi.RLock()
  234. res = len(s.items)
  235. s.mi.RUnlock()
  236. return
  237. }
  238. func (s *CacheMap) Keys() (res []any) {
  239. s.mi.RLock()
  240. res = make([]any, 0, len(s.items))
  241. for key := range s.items {
  242. res = append(res, key)
  243. }
  244. s.mi.RUnlock()
  245. return
  246. }
  247. func (s *CacheMap) Clear() {
  248. s.mi.Lock()
  249. s.items = make(map[any]*item)
  250. s.mi.Unlock()
  251. }