Skip to content

Commit 30f7ed7

Browse files
committed
Start work on disk cache for choose files preview caching
1 parent ec8ac5d commit 30f7ed7

File tree

2 files changed

+323
-0
lines changed

2 files changed

+323
-0
lines changed

tools/disk_cache/api.go

Lines changed: 82 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,82 @@
1+
package disk_cache
2+
3+
import (
4+
"crypto/sha256"
5+
"encoding/hex"
6+
"fmt"
7+
"os"
8+
"path/filepath"
9+
"sync"
10+
"time"
11+
12+
"github.com/kovidgoyal/kitty/tools/utils"
13+
)
14+
15+
var _ = fmt.Print
16+
17+
type Entry struct {
18+
Key string
19+
Size int64
20+
LastUsed time.Time
21+
}
22+
23+
type Metadata struct {
24+
TotalSize int64
25+
SortedEntries []*Entry
26+
}
27+
28+
type DiskCache struct {
29+
Path string
30+
MaxSize int64
31+
32+
lock_file *os.File
33+
lock_mutex sync.Mutex
34+
entries Metadata
35+
entry_map map[string]*Entry
36+
entries_mod_time time.Time
37+
}
38+
39+
func NewDiskCache(path string, max_size int64) (dc *DiskCache, err error) {
40+
if path, err = filepath.Abs(path); err != nil {
41+
return
42+
}
43+
if err = os.MkdirAll(path, 0o700); err != nil {
44+
return
45+
}
46+
return &DiskCache{Path: path, MaxSize: max_size}, nil
47+
}
48+
49+
func KeyForPath(path string) (key string, err error) {
50+
if path, err = filepath.EvalSymlinks(path); err != nil {
51+
return
52+
}
53+
if path, err = filepath.Abs(path); err != nil {
54+
return
55+
}
56+
57+
s, err := os.Stat(path)
58+
if err != nil {
59+
return
60+
}
61+
data := fmt.Sprintf("%s\x00%d\x00%d", path, s.Size(), s.ModTime().UnixNano())
62+
sum := sha256.Sum256(utils.UnsafeStringToBytes(data))
63+
return hex.EncodeToString(sum[:]), nil
64+
}
65+
66+
func (dc *DiskCache) Get(key string, items ...string) map[string]string {
67+
dc.lock()
68+
defer dc.unlock()
69+
return dc.get(key, items)
70+
}
71+
72+
func (dc *DiskCache) Remove(key string) (err error) {
73+
dc.lock()
74+
defer dc.unlock()
75+
return dc.remove(key)
76+
}
77+
78+
func (dc *DiskCache) Add(key string, items map[string][]byte) (err error) {
79+
dc.lock()
80+
defer dc.unlock()
81+
return dc.add(key, items)
82+
}

tools/disk_cache/implementation.go

Lines changed: 241 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,241 @@
1+
package disk_cache
2+
3+
import (
4+
"bytes"
5+
"encoding/json"
6+
"fmt"
7+
"os"
8+
"path/filepath"
9+
"slices"
10+
"time"
11+
12+
"github.com/kovidgoyal/kitty/tools/utils"
13+
)
14+
15+
var _ = fmt.Print
16+
17+
func (dc *DiskCache) lock() (err error) {
18+
dc.lock_mutex.Lock()
19+
defer dc.lock_mutex.Unlock()
20+
if dc.lock_file != nil {
21+
return
22+
}
23+
if dc.lock_file, err = os.OpenFile(filepath.Join(dc.Path, "lockfile"), os.O_RDWR|os.O_CREATE, 0o600); err != nil {
24+
return
25+
}
26+
return utils.LockFileExclusive(dc.lock_file)
27+
}
28+
29+
func (dc *DiskCache) unlock() {
30+
dc.lock_mutex.Lock()
31+
defer dc.lock_mutex.Unlock()
32+
if dc.lock_file != nil {
33+
utils.UnlockFile(dc.lock_file)
34+
dc.lock_file.Close()
35+
dc.lock_file = nil
36+
}
37+
}
38+
39+
func (dc *DiskCache) entries_path() string { return filepath.Join(dc.Path, "entries.json") }
40+
41+
func (dc *DiskCache) write_entries() (err error) {
42+
if d, err := json.Marshal(dc.entries); err != nil {
43+
return err
44+
} else {
45+
return os.WriteFile(dc.entries_path(), d, 0o600)
46+
}
47+
}
48+
49+
func (dc *DiskCache) rebuild_entries() error {
50+
if entries, err := os.ReadDir(dc.Path); err != nil {
51+
return err
52+
} else {
53+
ans := make(map[string]*Entry)
54+
var total int64
55+
for _, x := range entries {
56+
if x.IsDir() {
57+
if sub_entries, err := os.ReadDir(filepath.Join(dc.Path, x.Name())); err == nil && len(sub_entries) == 1 {
58+
key := sub_entries[0].Name()
59+
path := dc.folder_for_key(key)
60+
if file_entries, err := os.ReadDir(path); err == nil {
61+
e := Entry{}
62+
for _, f := range file_entries {
63+
if fi, err := f.Info(); err == nil {
64+
e.Size += fi.Size()
65+
if fi.ModTime().After(e.LastUsed) {
66+
e.LastUsed = fi.ModTime()
67+
}
68+
}
69+
}
70+
ans[key] = &e
71+
total += e.Size
72+
}
73+
}
74+
}
75+
}
76+
sorted := utils.Values(ans)
77+
slices.SortFunc(sorted, func(a, b *Entry) int {
78+
return a.LastUsed.Compare(b.LastUsed)
79+
})
80+
dc.entries = Metadata{TotalSize: total, SortedEntries: sorted}
81+
dc.entry_map = ans
82+
}
83+
return nil
84+
}
85+
86+
func (dc *DiskCache) ensure_entries() error {
87+
needed := dc.entry_map == nil
88+
path := dc.entries_path()
89+
if !needed {
90+
if s, err := os.Stat(path); err == nil && s.ModTime().After(dc.entries_mod_time) {
91+
needed = true
92+
}
93+
}
94+
if needed {
95+
if data, err := os.ReadFile(path); err != nil {
96+
if os.IsNotExist(err) {
97+
dc.entry_map = make(map[string]*Entry)
98+
dc.entries = Metadata{SortedEntries: make([]*Entry, 0)}
99+
} else {
100+
return err
101+
}
102+
} else {
103+
dc.entries = Metadata{SortedEntries: make([]*Entry, 0)}
104+
if err := json.Unmarshal(data, &dc.entries); err != nil {
105+
// corrupted data
106+
dc.rebuild_entries()
107+
}
108+
dc.entry_map = make(map[string]*Entry)
109+
for _, e := range dc.entries.SortedEntries {
110+
dc.entry_map[e.Key] = e
111+
}
112+
}
113+
}
114+
return nil
115+
}
116+
117+
func (dc *DiskCache) folder_for_key(key string) (ans string) {
118+
if len(key) < 5 {
119+
ans = filepath.Join(key, key)
120+
} else {
121+
ans = filepath.Join(key[:4], key)
122+
}
123+
return filepath.Join(dc.Path, ans)
124+
}
125+
126+
func (dc *DiskCache) update_last_used(key string) {
127+
if dc.ensure_entries() == nil {
128+
dc.update_timestamp(key)
129+
}
130+
131+
}
132+
133+
func (dc *DiskCache) get(key string, items []string) map[string]string {
134+
ans := make(map[string]string, len(items))
135+
base := dc.folder_for_key(key)
136+
if s, err := os.Stat(base); err != nil || !s.IsDir() {
137+
return ans
138+
}
139+
for _, x := range items {
140+
p := filepath.Join(base, x)
141+
if s, err := os.Stat(p); err != nil || !s.IsDir() {
142+
continue
143+
}
144+
ans[x] = p
145+
}
146+
dc.update_last_used(key)
147+
return ans
148+
}
149+
150+
func (dc *DiskCache) remove(key string) (err error) {
151+
if err = dc.ensure_entries(); err != nil {
152+
return
153+
}
154+
base := dc.folder_for_key(key)
155+
if err = os.RemoveAll(base); err == nil {
156+
t := dc.entry_map[key]
157+
if t != nil {
158+
delete(dc.entry_map, key)
159+
dc.entries.TotalSize = max(0, dc.entries.TotalSize-t.Size)
160+
dc.entries.SortedEntries = utils.Filter(dc.entries.SortedEntries, func(x *Entry) bool { return x.Key != key })
161+
return dc.write_entries()
162+
}
163+
}
164+
return
165+
}
166+
167+
func (dc *DiskCache) prune() error {
168+
if dc.entries.TotalSize <= dc.MaxSize {
169+
return nil
170+
}
171+
for dc.entries.TotalSize > dc.MaxSize && len(dc.entries.SortedEntries) > 0 {
172+
base := dc.folder_for_key(dc.entries.SortedEntries[0].Key)
173+
if err := os.RemoveAll(base); err == nil {
174+
t := dc.entries.SortedEntries[0]
175+
delete(dc.entry_map, t.Key)
176+
dc.entries.TotalSize = max(0, dc.entries.TotalSize-t.Size)
177+
dc.entries.SortedEntries = dc.entries.SortedEntries[1:]
178+
} else {
179+
return err
180+
}
181+
}
182+
return nil
183+
}
184+
185+
func (dc *DiskCache) update_timestamp(key string) {
186+
t := dc.entry_map[key]
187+
t.LastUsed = time.Now()
188+
idx := slices.Index(dc.entries.SortedEntries, t)
189+
copy(dc.entries.SortedEntries[idx:], dc.entries.SortedEntries[idx+1:])
190+
dc.entries.SortedEntries[len(dc.entries.SortedEntries)-1] = t
191+
}
192+
193+
func (dc *DiskCache) update_accounting(key string, changed int64) (err error) {
194+
if err = dc.ensure_entries(); err == nil {
195+
t := dc.entry_map[key]
196+
old_size := t.Size
197+
t.Size += changed
198+
t.Size = max(0, t.Size)
199+
dc.entries.TotalSize += t.Size - old_size
200+
dc.update_timestamp(key)
201+
dc.prune()
202+
return dc.write_entries()
203+
}
204+
return
205+
}
206+
207+
func (dc *DiskCache) add(key string, items map[string][]byte) (err error) {
208+
if err = dc.ensure_entries(); err != nil {
209+
return
210+
}
211+
base := dc.folder_for_key(key)
212+
if err = os.MkdirAll(base, 0o700); err != nil {
213+
return err
214+
}
215+
var changed int64
216+
defer func() {
217+
e := dc.update_accounting(key, changed)
218+
if err == nil {
219+
err = e
220+
}
221+
}()
222+
for x, data := range items {
223+
p := filepath.Join(base, x)
224+
var before int64
225+
if s, err := os.Stat(p); err == nil {
226+
before = s.Size()
227+
}
228+
if len(data) == 0 {
229+
if err = os.Remove(p); err != nil {
230+
return
231+
}
232+
changed -= before
233+
} else {
234+
if err = utils.AtomicWriteFile(p, bytes.NewReader(data), 0o700); err != nil {
235+
return
236+
}
237+
changed += int64(len(data)) - before
238+
}
239+
}
240+
return
241+
}

0 commit comments

Comments
 (0)