2017-11-06 21:38:52 +00:00
|
|
|
package vfs
|
|
|
|
|
|
|
|
import (
|
2017-11-28 14:18:48 +00:00
|
|
|
"fmt"
|
2017-11-06 21:38:52 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2017-11-28 14:18:48 +00:00
|
|
|
"sort"
|
|
|
|
"strings"
|
2017-11-06 21:38:52 +00:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/djherbis/times"
|
|
|
|
"github.com/ncw/rclone/fstest"
|
|
|
|
"github.com/spf13/pflag"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"golang.org/x/net/context" // switch to "context" when we stop supporting go1.6
|
|
|
|
)
|
|
|
|
|
|
|
|
// Check CacheMode it satisfies the pflag interface
|
|
|
|
var _ pflag.Value = (*CacheMode)(nil)
|
|
|
|
|
|
|
|
func TestCacheModeString(t *testing.T) {
|
|
|
|
assert.Equal(t, "off", CacheModeOff.String())
|
|
|
|
assert.Equal(t, "full", CacheModeFull.String())
|
|
|
|
assert.Equal(t, "CacheMode(17)", CacheMode(17).String())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCacheModeSet(t *testing.T) {
|
|
|
|
var m CacheMode
|
|
|
|
|
|
|
|
err := m.Set("full")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, CacheModeFull, m)
|
|
|
|
|
|
|
|
err = m.Set("potato")
|
|
|
|
assert.Error(t, err, "Unknown cache mode level")
|
|
|
|
|
|
|
|
err = m.Set("")
|
|
|
|
assert.Error(t, err, "Unknown cache mode level")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCacheModeType(t *testing.T) {
|
|
|
|
var m CacheMode
|
|
|
|
assert.Equal(t, "string", m.Type())
|
|
|
|
}
|
|
|
|
|
2017-11-28 14:18:48 +00:00
|
|
|
// convert c.item to a string
|
|
|
|
func itemAsString(c *cache) string {
|
|
|
|
c.itemMu.Lock()
|
|
|
|
defer c.itemMu.Unlock()
|
|
|
|
var out []string
|
|
|
|
for name, item := range c.item {
|
|
|
|
out = append(out, fmt.Sprintf("name=%q isFile=%v opens=%d", name, item.isFile, item.opens))
|
|
|
|
}
|
|
|
|
sort.Strings(out)
|
|
|
|
return strings.Join(out, "\n")
|
|
|
|
}
|
|
|
|
|
2017-11-06 21:38:52 +00:00
|
|
|
func TestCacheNew(t *testing.T) {
|
|
|
|
r := fstest.NewRun(t)
|
|
|
|
defer r.Finalise()
|
|
|
|
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
|
|
|
|
c, err := newCache(ctx, r.Fremote, &DefaultOpt)
|
2018-02-16 12:05:59 +00:00
|
|
|
cancel() // kill the background cache cleaning as it interferes with the tests
|
2017-11-06 21:38:52 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Contains(t, c.root, "vfs")
|
|
|
|
assert.Contains(t, c.f.Root(), filepath.Base(r.Fremote.Root()))
|
2017-11-28 14:18:48 +00:00
|
|
|
assert.Equal(t, "", itemAsString(c))
|
2017-11-06 21:38:52 +00:00
|
|
|
|
|
|
|
// mkdir
|
|
|
|
p, err := c.mkdir("potato")
|
|
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, "potato", filepath.Base(p))
|
2017-11-28 14:18:48 +00:00
|
|
|
assert.Equal(t, `name="" isFile=false opens=0`, itemAsString(c))
|
2017-11-06 21:38:52 +00:00
|
|
|
|
|
|
|
fi, err := os.Stat(filepath.Dir(p))
|
|
|
|
require.NoError(t, err)
|
|
|
|
assert.True(t, fi.IsDir())
|
|
|
|
|
|
|
|
// get
|
|
|
|
item := c.get("potato")
|
|
|
|
item2 := c.get("potato")
|
|
|
|
assert.Equal(t, item, item2)
|
|
|
|
assert.WithinDuration(t, time.Now(), item.atime, time.Second)
|
|
|
|
|
|
|
|
// updateTime
|
|
|
|
//.. before
|
|
|
|
t1 := time.Now().Add(-60 * time.Minute)
|
|
|
|
c.updateTime("potato", t1)
|
|
|
|
item = c.get("potato")
|
|
|
|
assert.NotEqual(t, t1, item.atime)
|
|
|
|
assert.Equal(t, 0, item.opens)
|
|
|
|
//..after
|
|
|
|
t2 := time.Now().Add(60 * time.Minute)
|
|
|
|
c.updateTime("potato", t2)
|
|
|
|
item = c.get("potato")
|
|
|
|
assert.Equal(t, t2, item.atime)
|
|
|
|
assert.Equal(t, 0, item.opens)
|
|
|
|
|
|
|
|
// open
|
2017-11-28 14:18:48 +00:00
|
|
|
assert.Equal(t, `name="" isFile=false opens=0
|
|
|
|
name="potato" isFile=true opens=0`, itemAsString(c))
|
2017-11-06 21:38:52 +00:00
|
|
|
c.open("potato")
|
2017-11-28 14:18:48 +00:00
|
|
|
assert.Equal(t, `name="" isFile=false opens=1
|
|
|
|
name="potato" isFile=true opens=1`, itemAsString(c))
|
2017-11-06 21:38:52 +00:00
|
|
|
item = c.get("potato")
|
|
|
|
assert.WithinDuration(t, time.Now(), item.atime, time.Second)
|
|
|
|
assert.Equal(t, 1, item.opens)
|
|
|
|
|
|
|
|
// write the file
|
|
|
|
err = ioutil.WriteFile(p, []byte("hello"), 0600)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// read its atime
|
|
|
|
fi, err = os.Stat(p)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
atime := times.Get(fi).AccessTime()
|
|
|
|
|
|
|
|
// updateAtimes
|
|
|
|
item = c.get("potato")
|
|
|
|
item.atime = time.Now().Add(-24 * time.Hour)
|
|
|
|
err = c.updateAtimes()
|
|
|
|
require.NoError(t, err)
|
2017-11-28 14:18:48 +00:00
|
|
|
assert.Equal(t, `name="" isFile=false opens=1
|
|
|
|
name="potato" isFile=true opens=1`, itemAsString(c))
|
2017-11-06 21:38:52 +00:00
|
|
|
item = c.get("potato")
|
|
|
|
assert.Equal(t, atime, item.atime)
|
|
|
|
|
2018-02-02 12:06:42 +00:00
|
|
|
// updateAtimes - not in the cache
|
|
|
|
oldItem := item
|
2018-02-15 21:34:37 +00:00
|
|
|
c.itemMu.Lock()
|
2018-02-02 12:06:42 +00:00
|
|
|
delete(c.item, "potato") // remove from cache
|
2018-02-15 21:34:37 +00:00
|
|
|
c.itemMu.Unlock()
|
2018-02-02 12:06:42 +00:00
|
|
|
err = c.updateAtimes()
|
|
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=1
|
|
|
|
name="potato" isFile=true opens=0`, itemAsString(c))
|
|
|
|
item = c.get("potato")
|
|
|
|
assert.Equal(t, atime, item.atime)
|
2018-02-15 21:34:37 +00:00
|
|
|
c.itemMu.Lock()
|
2018-02-02 12:06:42 +00:00
|
|
|
c.item["potato"] = oldItem // restore to cache
|
2018-02-15 21:34:37 +00:00
|
|
|
c.itemMu.Unlock()
|
2018-02-02 12:06:42 +00:00
|
|
|
|
2017-11-06 21:38:52 +00:00
|
|
|
// try purging with file open
|
|
|
|
c.purgeOld(10 * time.Second)
|
|
|
|
_, err = os.Stat(p)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
// close
|
2017-11-28 14:18:48 +00:00
|
|
|
assert.Equal(t, `name="" isFile=false opens=1
|
|
|
|
name="potato" isFile=true opens=1`, itemAsString(c))
|
2017-11-06 21:38:52 +00:00
|
|
|
c.updateTime("potato", t2)
|
2017-11-28 14:18:48 +00:00
|
|
|
assert.Equal(t, `name="" isFile=false opens=1
|
|
|
|
name="potato" isFile=true opens=1`, itemAsString(c))
|
2017-11-06 21:38:52 +00:00
|
|
|
c.close("potato")
|
2017-11-28 14:18:48 +00:00
|
|
|
assert.Equal(t, `name="" isFile=false opens=0
|
|
|
|
name="potato" isFile=true opens=0`, itemAsString(c))
|
2017-11-06 21:38:52 +00:00
|
|
|
item = c.get("potato")
|
|
|
|
assert.WithinDuration(t, time.Now(), item.atime, time.Second)
|
|
|
|
assert.Equal(t, 0, item.opens)
|
|
|
|
|
|
|
|
// try purging with file closed
|
|
|
|
c.purgeOld(10 * time.Second)
|
|
|
|
// ...nothing should happend
|
|
|
|
_, err = os.Stat(p)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
//.. purge again with -ve age
|
|
|
|
c.purgeOld(-10 * time.Second)
|
|
|
|
_, err = os.Stat(p)
|
|
|
|
assert.True(t, os.IsNotExist(err))
|
|
|
|
|
|
|
|
// clean - have tested the internals already
|
|
|
|
c.clean()
|
|
|
|
|
|
|
|
// cleanup
|
|
|
|
err = c.cleanUp()
|
|
|
|
require.NoError(t, err)
|
|
|
|
_, err = os.Stat(c.root)
|
|
|
|
assert.True(t, os.IsNotExist(err))
|
|
|
|
}
|
2017-11-28 14:18:48 +00:00
|
|
|
|
|
|
|
func TestCacheOpens(t *testing.T) {
|
|
|
|
r := fstest.NewRun(t)
|
|
|
|
defer r.Finalise()
|
|
|
|
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
c, err := newCache(ctx, r.Fremote, &DefaultOpt)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Equal(t, "", itemAsString(c))
|
|
|
|
c.open("potato")
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=1
|
|
|
|
name="potato" isFile=true opens=1`, itemAsString(c))
|
|
|
|
c.open("potato")
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=2
|
|
|
|
name="potato" isFile=true opens=2`, itemAsString(c))
|
|
|
|
c.close("potato")
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=1
|
|
|
|
name="potato" isFile=true opens=1`, itemAsString(c))
|
|
|
|
c.close("potato")
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=0
|
|
|
|
name="potato" isFile=true opens=0`, itemAsString(c))
|
|
|
|
|
|
|
|
c.open("potato")
|
|
|
|
c.open("a/b/c/d/one")
|
|
|
|
c.open("a/b/c/d/e/two")
|
|
|
|
c.open("a/b/c/d/e/f/three")
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=4
|
|
|
|
name="a" isFile=false opens=3
|
|
|
|
name="a/b" isFile=false opens=3
|
|
|
|
name="a/b/c" isFile=false opens=3
|
|
|
|
name="a/b/c/d" isFile=false opens=3
|
|
|
|
name="a/b/c/d/e" isFile=false opens=2
|
|
|
|
name="a/b/c/d/e/f" isFile=false opens=1
|
|
|
|
name="a/b/c/d/e/f/three" isFile=true opens=1
|
|
|
|
name="a/b/c/d/e/two" isFile=true opens=1
|
|
|
|
name="a/b/c/d/one" isFile=true opens=1
|
|
|
|
name="potato" isFile=true opens=1`, itemAsString(c))
|
|
|
|
|
|
|
|
c.close("potato")
|
|
|
|
c.close("a/b/c/d/one")
|
|
|
|
c.close("a/b/c/d/e/two")
|
|
|
|
c.close("a/b/c/d/e/f/three")
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=0
|
|
|
|
name="a" isFile=false opens=0
|
|
|
|
name="a/b" isFile=false opens=0
|
|
|
|
name="a/b/c" isFile=false opens=0
|
|
|
|
name="a/b/c/d" isFile=false opens=0
|
|
|
|
name="a/b/c/d/e" isFile=false opens=0
|
|
|
|
name="a/b/c/d/e/f" isFile=false opens=0
|
|
|
|
name="a/b/c/d/e/f/three" isFile=true opens=0
|
|
|
|
name="a/b/c/d/e/two" isFile=true opens=0
|
|
|
|
name="a/b/c/d/one" isFile=true opens=0
|
|
|
|
name="potato" isFile=true opens=0`, itemAsString(c))
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCacheCacheDir(t *testing.T) {
|
|
|
|
r := fstest.NewRun(t)
|
|
|
|
defer r.Finalise()
|
|
|
|
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
c, err := newCache(ctx, r.Fremote, &DefaultOpt)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Equal(t, "", itemAsString(c))
|
|
|
|
|
|
|
|
c.cacheDir("dir")
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=0
|
|
|
|
name="dir" isFile=false opens=0`, itemAsString(c))
|
|
|
|
|
|
|
|
c.cacheDir("dir/sub")
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=0
|
|
|
|
name="dir" isFile=false opens=0
|
|
|
|
name="dir/sub" isFile=false opens=0`, itemAsString(c))
|
|
|
|
|
|
|
|
c.cacheDir("dir/sub2/subsub2")
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=0
|
|
|
|
name="dir" isFile=false opens=0
|
|
|
|
name="dir/sub" isFile=false opens=0
|
|
|
|
name="dir/sub2" isFile=false opens=0
|
|
|
|
name="dir/sub2/subsub2" isFile=false opens=0`, itemAsString(c))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCachePurgeOld(t *testing.T) {
|
|
|
|
r := fstest.NewRun(t)
|
|
|
|
defer r.Finalise()
|
|
|
|
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
c, err := newCache(ctx, r.Fremote, &DefaultOpt)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Test funcs
|
|
|
|
var removed []string
|
|
|
|
removedDir := true
|
|
|
|
removeFile := func(name string) {
|
|
|
|
removed = append(removed, name)
|
|
|
|
}
|
|
|
|
removeDir := func(name string) bool {
|
|
|
|
if removedDir {
|
|
|
|
removed = append(removed, name+"/")
|
|
|
|
}
|
|
|
|
return removedDir
|
|
|
|
}
|
|
|
|
|
|
|
|
removed = nil
|
|
|
|
c._purgeOld(-10*time.Second, removeFile, removeDir)
|
|
|
|
assert.Equal(t, []string(nil), removed)
|
|
|
|
|
|
|
|
c.open("sub/dir/potato")
|
|
|
|
c.close("sub/dir/potato")
|
|
|
|
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=0
|
|
|
|
name="sub" isFile=false opens=0
|
|
|
|
name="sub/dir" isFile=false opens=0
|
|
|
|
name="sub/dir/potato" isFile=true opens=0`, itemAsString(c))
|
|
|
|
|
|
|
|
removed = nil
|
|
|
|
removedDir = false
|
|
|
|
c._purgeOld(10*time.Second, removeFile, removeDir)
|
|
|
|
assert.Equal(t, []string(nil), removed)
|
|
|
|
|
|
|
|
assert.Equal(t, `name="" isFile=false opens=0
|
|
|
|
name="sub" isFile=false opens=0
|
|
|
|
name="sub/dir" isFile=false opens=0
|
|
|
|
name="sub/dir/potato" isFile=true opens=0`, itemAsString(c))
|
|
|
|
|
|
|
|
removed = nil
|
|
|
|
removedDir = true
|
|
|
|
c._purgeOld(-10*time.Second, removeFile, removeDir)
|
|
|
|
assert.Equal(t, []string{
|
|
|
|
"sub/dir/potato",
|
|
|
|
"sub/dir/",
|
|
|
|
"sub/",
|
|
|
|
"/",
|
|
|
|
}, removed)
|
|
|
|
|
|
|
|
assert.Equal(t, ``, itemAsString(c))
|
|
|
|
}
|