Files
kopia/fs/cachefs/cachefs.go
2018-04-03 17:39:54 -07:00

71 lines
1.4 KiB
Go

// Package cachefs implements a wrapper that caches filesystem actions.
package cachefs
import (
"context"
"github.com/kopia/kopia/fs"
)
// DirectoryCacher reads and potentially caches directory entries for a given directory.
type DirectoryCacher interface {
Readdir(ctx context.Context, d fs.Directory) (fs.Entries, error)
}
type cacheContext struct {
cacher DirectoryCacher
}
type directory struct {
ctx *cacheContext
fs.Directory
}
func (d *directory) Readdir(ctx context.Context) (fs.Entries, error) {
entries, err := d.ctx.cacher.Readdir(ctx, d.Directory)
if err != nil {
return entries, err
}
wrapped := make(fs.Entries, len(entries))
for i, entry := range entries {
wrapped[i] = wrapWithContext(entry, d.ctx)
}
return wrapped, err
}
type file struct {
ctx *cacheContext
fs.File
}
type symlink struct {
ctx *cacheContext
fs.Symlink
}
// Wrap returns an Entry that wraps another Entry and caches directory reads.
func Wrap(e fs.Entry, cacher DirectoryCacher) fs.Entry {
return wrapWithContext(e, &cacheContext{cacher})
}
func wrapWithContext(e fs.Entry, opts *cacheContext) fs.Entry {
switch e := e.(type) {
case fs.Directory:
return fs.Directory(&directory{opts, e})
case fs.File:
return fs.File(&file{opts, e})
case fs.Symlink:
return fs.Symlink(&symlink{opts, e})
default:
return e
}
}
var _ fs.Directory = &directory{}
var _ fs.File = &file{}
var _ fs.Symlink = &symlink{}