Documentation ¶
Overview ¶
Example ¶
// create the cache, keys expire after 1 hour. c, err := New("./cache", 0755, time.Hour) if err != nil { log.Fatal(err.Error()) } // wipe the cache when done defer c.Clean() // Get() and it's streams can be called concurrently but just for example: for i := 0; i < 3; i++ { r, w, err := c.Get("stream") if err != nil { log.Fatal(err.Error()) } if w != nil { // a new stream, write to it. go func() { w.Write([]byte("hello world\n")) w.Close() }() } // the stream has started, read from it io.Copy(os.Stdout, r) r.Close() }
Output: hello world hello world hello world
Index ¶
- func Handler(c Cache, h http.Handler) http.Handler
- func ListenAndServe(c Cache, addr string) error
- type Cache
- func New(dir string, perms os.FileMode, expiry time.Duration) (Cache, error)
- func NewCache(fs FileSystem, grim Reaper) (Cache, error)
- func NewCacheWithHaunter(fs FileSystem, haunter Haunter) (Cache, error)
- func NewLayered(caches ...Cache) Cache
- func NewPartition(d Distributor) Cache
- func NewRemote(raddr string) Cache
- type CacheAccessor
- type Distributor
- type Entry
- type FileInfo
- type FileSystem
- type FileSystemStater
- type Haunter
- type LRUHaunter
- type ReadAtCloser
- type Reaper
- Bugs
Examples ¶
Constants ¶
This section is empty.
Variables ¶
This section is empty.
Functions ¶
func Handler ¶
Handler is a caching middle-ware for http Handlers. It responds to http requests via the passed http.Handler, and caches the response using the passed cache. The cache key for the request is the req.URL.String(). Note: It does not cache http headers. It is more efficient to set them yourself.
Example ¶
c, err := New("./server", 0700, 0) if err != nil { log.Fatal(err.Error()) } defer c.Clean() handler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { fmt.Fprintln(w, "Hello Client") }) ts := httptest.NewServer(Handler(c, handler)) defer ts.Close() resp, err := http.Get(ts.URL) if err != nil { log.Fatal(err.Error()) } io.Copy(os.Stdout, resp.Body) resp.Body.Close()
Output: Hello Client
func ListenAndServe ¶
ListenAndServe hosts a Cache for access via NewRemote
Types ¶
type Cache ¶
type Cache interface { // Get manages access to the streams in the cache. // If the key does not exist, w != nil and you can start writing to the stream. // If the key does exist, w == nil. // r will always be non-nil as long as err == nil and you must close r when you're done reading. // Get can be called concurrently, and writing and reading is concurrent safe. Get(key string) (ReadAtCloser, io.WriteCloser, error) // Remove deletes the stream from the cache, blocking until the underlying // file can be deleted (all active streams finish with it). // It is safe to call Remove concurrently with Get. Remove(key string) error // Exists checks if a key is in the cache. // It is safe to call Exists concurrently with Get. Exists(key string) bool // Clean will empty the cache and delete the cache folder. // Clean is not safe to call while streams are being read/written. Clean() error }
Cache works like a concurrent-safe map for streams.
func New ¶
New creates a new Cache using NewFs(dir, perms). expiry is the duration after which an un-accessed key will be removed from the cache, a zero value expiro means never expire.
func NewCache ¶
func NewCache(fs FileSystem, grim Reaper) (Cache, error)
NewCache creates a new Cache based on FileSystem fs. fs.Files() are loaded using the name they were created with as a key. Reaper is used to determine when files expire, nil means never expire.
func NewCacheWithHaunter ¶ added in v0.9.0
func NewCacheWithHaunter(fs FileSystem, haunter Haunter) (Cache, error)
NewCacheWithHaunter create a new Cache based on FileSystem fs. fs.Files() are loaded using the name they were created with as a key. Haunter is used to determine when files expire, nil means never expire.
func NewLayered ¶
NewLayered returns a Cache which stores its data in all the passed caches, when a key is requested it is loaded into all the caches above the first hit.
func NewPartition ¶
func NewPartition(d Distributor) Cache
NewPartition returns a Cache which uses the Caches defined by the passed Distributor.
type CacheAccessor ¶ added in v0.9.0
type CacheAccessor interface { FileSystemStater EnumerateEntries(enumerator func(key string, e Entry) bool) RemoveFile(key string) }
type Distributor ¶
type Distributor interface { // GetCache will always return the same Cache for the same key. GetCache(key string) Cache // Clean should wipe all the caches this Distributor manages Clean() error }
Distributor provides a way to partition keys into Caches.
func NewDistributor ¶
func NewDistributor(caches ...Cache) Distributor
NewDistributor returns a Distributor which evenly distributes the keyspace into the passed caches.
type FileInfo ¶ added in v0.9.0
func (*FileInfo) AccessTime ¶ added in v0.9.0
AccessTime returns the last time the file was read. It will be used to check expiry of a file, and must be concurrent safe with modifications to the FileSystem (writes, reads etc.)
type FileSystem ¶
type FileSystem interface { // Stream FileSystem stream.FileSystem FileSystemStater // Reload should look through the FileSystem and call the supplied fn // with the key/filename pairs that are found. Reload(func(key, name string)) error // RemoveAll should empty the FileSystem of all files. RemoveAll() error }
FileSystem is used as the source for a Cache.
func NewFs ¶
func NewFs(dir string, mode os.FileMode) (FileSystem, error)
NewFs returns a FileSystem rooted at directory dir. Dir is created with perms if it doesn't exist.
func NewMemFs ¶
func NewMemFs() FileSystem
NewMemFs creates an in-memory FileSystem. It does not support persistence (Reload is a nop).
type FileSystemStater ¶ added in v0.9.0
type Haunter ¶ added in v0.9.0
type Haunter interface { Haunt(c CacheAccessor) Next() time.Duration }
func NewLRUHaunterStrategy ¶ added in v0.9.0
func NewLRUHaunterStrategy(haunter LRUHaunter) Haunter
NewLRUHaunterStrategy returns a simple scheduleHaunt which provides an implementation LRUHaunter strategy
func NewReaperHaunterStrategy ¶ added in v0.9.0
NewReaperHaunterStrategy returns a simple scheduleHaunt which provides an implementation Reaper strategy
type LRUHaunter ¶ added in v0.9.0
type LRUHaunter interface { // Returns the amount of time to wait before the next scheduled Reaping. Next() time.Duration // Given a CacheAccessor, return keys to reap list. Scrub(c CacheAccessor) []string }
LRUHaunter is used to control when there are too many streams or the size of the streams is too big. It is called once right after loading, and then it is run again after every Next() period of time.
func NewLRUHaunter ¶ added in v0.9.0
func NewLRUHaunter(maxItems int, maxSize int64, period time.Duration) LRUHaunter
NewLRUHaunter returns a simple haunter which runs every "period" and scrubs older files when the total file size is over maxSize or total item count is over maxItems. If maxItems or maxSize are 0, they won't be checked
type ReadAtCloser ¶ added in v0.8.0
type ReadAtCloser interface { io.ReadCloser io.ReaderAt }
ReadAtCloser is an io.ReadCloser, and an io.ReaderAt. It supports both so that Range Requests are possible.
type Reaper ¶
type Reaper interface { // Returns the amount of time to wait before the next scheduled Reaping. Next() time.Duration // Given a key and the last r/w times of a file, return true // to remove the file from the cache, false to keep it. Reap(key string, lastRead, lastWrite time.Time) bool }
Reaper is used to control when streams expire from the cache. It is called once right after loading, and then it is run again after every Next() period of time.
Notes ¶
Bugs ¶
Return an error if cleaning fails