forked from minotar/imgd
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcache_memory.go
103 lines (85 loc) · 2.57 KB
/
cache_memory.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
package main
import (
"github.com/minotar/minecraft"
"time"
)
const (
// Get the skin size in bytes. Stored as a []uint8, one byte each,
// plus bounces. So 64 * 64 bytes and we'll throw in an extra 16
// bytes of overhead.
SKIN_SIZE = (64 * 64) + 16
// Define a 64 MB cache size.
CACHE_SIZE = 2 << 25
// Based off those, calculate the maximum number of skins we'll store
// in memory.
SKIN_NUMBER = CACHE_SIZE / SKIN_SIZE
)
// Cache object that stores skins in memory.
type CacheMemory struct {
// Map of usernames to minecraft skins. Lookups here are O(1), so that
// makes my happy.
Skins map[string]minecraft.Skin
// Additionally keep a *slice* of usernames which we can update
Usernames []string
}
// Find the position of a string in a slice. Returns -1 on failure.
func indexOf(str string, list []string) int {
for index, value := range list {
if value == str {
return index
}
}
return -1
}
func (c *CacheMemory) setup() error {
c.Skins = map[string]minecraft.Skin{}
c.Usernames = []string{}
log.Info("Loaded Memory cache")
return nil
}
// Returns whether the item exists in the cache.
func (c *CacheMemory) has(username string) bool {
if _, exists := c.Skins[username]; exists {
return true
} else {
return false
}
}
// Retrieves the item from the cache. We'll promote it to the "top" of the
// cache, effectively updating its expiry time.
func (c *CacheMemory) pull(username string) minecraft.Skin {
index := indexOf(username, c.Usernames)
c.Usernames = append(c.Usernames, username)
c.Usernames = append(c.Usernames[:index], c.Usernames[index+1:]...)
return c.Skins[username]
}
// Removes the username from the cache
func (c *CacheMemory) remove(username string) {
index := indexOf(username, c.Usernames)
if index == -1 {
return
}
key := c.Usernames[index]
delete(c.Skins, key)
}
// Adds the skin to the cache, remove the oldest, expired skin if the cache
// list is full.
func (c *CacheMemory) add(username string, skin minecraft.Skin) {
if len(c.Usernames) >= SKIN_NUMBER {
first := c.Usernames[0]
delete(c.Skins, first)
c.Usernames = append(c.Usernames[1:], username)
} else {
c.Usernames = append(c.Usernames, username)
}
c.Skins[username] = skin
// After the expiration time, remove the item from the cache.
time.AfterFunc(time.Duration(config.Server.Ttl)*time.Second, func() {
c.remove(username)
})
}
// The byte size of the cache. Fairly rough... don't really want to venture
// into the land of manual memory management, because there be dragons.
func (c *CacheMemory) memory() uint64 {
return uint64(len(c.Usernames) * SKIN_SIZE)
}