Skip to content

Commit

Permalink
Add a file cache
Browse files Browse the repository at this point in the history
  • Loading branch information
bep committed Nov 8, 2018
1 parent 2900801 commit 6626292
Show file tree
Hide file tree
Showing 2 changed files with 326 additions and 0 deletions.
163 changes: 163 additions & 0 deletions cache/filecache/filecache.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,163 @@
// Copyright 2018 The Hugo Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package filecache

import (
"io"
"path/filepath"
"strings"
"time"

"github.com/BurntSushi/locker"
"github.com/bep/mapstructure"
"github.com/gohugoio/hugo/common/hugio"
"github.com/gohugoio/hugo/config"
"github.com/spf13/afero"
)

const cachesConfigKey = "caches"

var defaultCacheConfig = cacheConfig{
TTL: -1,
}

type cachesConfig map[string]cacheConfig

type cacheConfig struct {
// Time to Live. Any items older than this will be removed and
// not returned from the cache.
// -1 means forever.
TTL int

// The directory where files are stored.
Dir string
}

// CI cache dir provider
//

// FileCache caches a set of files in a directory. This is usually a file on
// disk, but since this is backed by an Afero file system, it can be anything.
type FileCache struct {
fs afero.Fs

ttl time.Duration

nlocker *locker.Locker
}

// Get gets a file from the cache given a filename. It will return nil
// if file was not found in cache or if it's expired.
func (c *FileCache) Get(filename string) hugio.ReadSeekCloser {
filename = filepath.Clean(filename)

c.nlocker.RLock(filename)

fi, err := c.fs.Stat(filename)
if err != nil {
c.nlocker.RUnlock(filename)
return nil
}

expiry := time.Now().Add(-c.ttl)
expired := fi.ModTime().Before(expiry)

if expired {
// Need a write lock for this.
c.nlocker.RUnlock(filename)
c.nlocker.Lock(filename)

// Double check
fi, err := c.fs.Stat(filename)
expired := err == nil && fi.ModTime().Before(expiry)
if expired {
c.fs.Remove(filename)
}

c.nlocker.Unlock(filename)

if err != nil || expired {
return nil
}

c.nlocker.RLock(filename)

}

defer c.nlocker.RUnlock(filename)

f, err := c.fs.Open(filename)
if err != nil {
return nil
}
return f
}

// WriteReader writes r to filename in the file cache.
func (c *FileCache) WriteReader(filename string, r io.Reader) error {
filename = filepath.Clean(filename)
c.nlocker.Lock(filename)
defer c.nlocker.Unlock(filename)

return afero.WriteReader(c.fs, filename, r)
}

type FileCaches map[string]*FileCache

// Get gets a named cache, nil if none found.
func (f FileCaches) Get(name string) *FileCache {
return f[strings.ToLower(name)]
}

// FileCaches holds a map of named file caches.
//type FileCaches map[string]*FileCache

// NewFileCachesFromConfig creates a new set of file caches from the given
// configuration.
func NewFileCachesFromConfig(fs afero.Fs, cfg config.Provider) (FileCaches, error) {
dcfg, err := decodeConfig(cfg)
if err != nil {
return nil, err
}
m := make(FileCaches)
for k, v := range dcfg {
m[k] = &FileCache{
fs: fs,
nlocker: locker.NewLocker(),
ttl: time.Second * time.Duration(v.TTL),
}
}
return m, nil
}

func decodeConfig(cfg config.Provider) (cachesConfig, error) {
c := make(cachesConfig)
if !cfg.IsSet(cachesConfigKey) {
return c, nil
}

m := cfg.GetStringMap(cachesConfigKey)

for k, v := range m {
cc := defaultCacheConfig

if err := mapstructure.WeakDecode(v, &cc); err != nil {
return nil, err
}

c[strings.ToLower(k)] = cc
}

return c, nil
}
163 changes: 163 additions & 0 deletions cache/filecache/filecache_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,163 @@
// Copyright 2018 The Hugo Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package filecache

import (
"fmt"
"io/ioutil"
"strings"
"sync"
"testing"
"time"

"github.com/gohugoio/hugo/config"

"github.com/spf13/afero"

"github.com/stretchr/testify/require"
)

func TestFileCache(t *testing.T) {
t.Parallel()
assert := require.New(t)

configStr := `
[caches]
[caches.concurrent]
ttl = 111
dir = "/cache/c"
`

cfg, err := config.FromConfigString(configStr, "toml")
assert.NoError(err)

caches, err := NewFileCachesFromConfig(afero.NewMemMapFs(), cfg)
assert.NoError(err)

const cacheName = "Concurrent"

c := caches.Get(cacheName)
assert.NotNil(c)
assert.Equal(time.Duration(111)*time.Second, c.ttl)

data := "abc"

assert.NoError(c.WriteReader("a", strings.NewReader(data)))

r := c.Get("a")
assert.NotNil(r)
b, _ := ioutil.ReadAll(r)
r.Close()
assert.Equal(data, string(b))

assert.NotNil(caches.Get(strings.ToUpper(cacheName)))

}

func TestFileCacheConcurrent(t *testing.T) {
t.Parallel()

assert := require.New(t)

configStr := `
[caches]
[caches.concurrent]
ttl = 111
dir = "/cache/c"
`

cfg, err := config.FromConfigString(configStr, "toml")
assert.NoError(err)

caches, err := NewFileCachesFromConfig(afero.NewMemMapFs(), cfg)
assert.NoError(err)

const cacheName = "concurrent"

filenameData := func(i int) (string, string) {
data := fmt.Sprintf("data: %d", i)
filename := fmt.Sprintf("file%d", i)
return filename, data
}

var wg sync.WaitGroup

for i := 0; i < 100; i++ {
wg.Add(1)
go func() {
defer wg.Done()
for j := 0; j < 10; j++ {
c := caches.Get(cacheName)
assert.NotNil(c)
filename, data := filenameData(i)
assert.NoError(c.WriteReader(filename, strings.NewReader(data)))
}
}()

wg.Add(1)
go func() {
defer wg.Done()
for j := 0; j < 10; j++ {
c := caches.Get(cacheName)
assert.NotNil(c)
filename, data := filenameData(i)
r := c.Get(filename)
if r != nil {
b, _ := ioutil.ReadAll(r)
r.Close()
assert.Equal(data, string(b))
}
}
}()
}
wg.Wait()
}

func TestDecodeConfig(t *testing.T) {
t.Parallel()

assert := require.New(t)

configStr := `
[caches]
[caches.c1]
ttl = 1234
dir = "/path/to/c1"
[caches.c2]
ttl = 3456
dir = "/path/to/c2"
[caches.c3]
dir = "/path/to/c3"
`

cfg, err := config.FromConfigString(configStr, "toml")
assert.NoError(err)

decoded, err := decodeConfig(cfg)
assert.NoError(err)

assert.Equal(3, len(decoded))

c2 := decoded["c2"]
assert.Equal(3456, c2.TTL)
assert.Equal("/path/to/c2", c2.Dir)

c3 := decoded["c3"]
assert.Equal(-1, c3.TTL)
assert.Equal("/path/to/c3", c3.Dir)

}

0 comments on commit 6626292

Please sign in to comment.