-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcache.go
113 lines (91 loc) · 1.96 KB
/
cache.go
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
package lcache
import (
"errors"
"sort"
"sync"
"time"
)
type Cache struct {
sync.Mutex
cache map[string]string
accessTime map[int64]string
size int
}
type CacheResult struct {
Value interface{}
Error error
}
var KeyNotFoundError = errors.New("key not found")
func NewCache(size ...int) *Cache { //make size optional
data := make(map[string]string)
timestamp := make(map[int64]string)
if len(size) == 0 {
return &Cache{
Mutex: sync.Mutex{},
cache: data,
accessTime: timestamp,
size: -1,
}
}
return &Cache{
Mutex: sync.Mutex{},
cache: data,
accessTime: timestamp,
size: size[0],
}
}
func (c *Cache) Read(key string) *CacheResult {
c.Lock()
defer c.Unlock()
var r *CacheResult
var value, keyExists = c.lookup(key)
if !keyExists {
return &CacheResult{
Value: nil,
Error: KeyNotFoundError,
}
}
r = &CacheResult{
Value: value,
Error: nil,
}
//update last access time
c.accessTime[time.Now().UnixNano()] = key
return r
}
func (c *Cache) Write(key, data string) {
c.Lock()
defer c.Unlock()
timestamp := time.Now().UnixNano()
if c.size == -1 || len(c.cache) < c.size { //we haven't reached Cache size
c.cache[key] = data
c.accessTime[timestamp] = key
return
}
if len(c.cache) == c.size {
lruKey, lruTime := c.getLeastRecentlyUsedKeys(c.accessTime)
delete(c.cache, lruKey)
delete(c.accessTime, lruTime)
}
//set or update
c.cache[key] = data
c.accessTime[timestamp] = key
return
}
func (c *Cache) lookup(key string) (v string, ok bool) {
v, ok = c.cache[key]
return
}
func (c *Cache) Size() int {
return len(c.cache)
}
func (c *Cache) getLeastRecentlyUsedKeys(timestampMap map[int64]string) (string, int64) {
var timeEntries []int64
for k, _ := range timestampMap {
timeEntries = append(timeEntries, k)
}
sort.Slice(timeEntries, func(i, j int) bool {
return timeEntries[i] < timeEntries[j]
})
return timestampMap[timeEntries[0]], timeEntries[0]
}