2020-02-29 19:08:22 +01:00
// Package vfscache deals with caching of files locally for the VFS layer
package vfscache
import (
"context"
"fmt"
"os"
"path"
"path/filepath"
"runtime"
"sort"
"strings"
"sync"
"time"
"github.com/pkg/errors"
"github.com/rclone/rclone/fs"
fscache "github.com/rclone/rclone/fs/cache"
"github.com/rclone/rclone/fs/config"
"github.com/rclone/rclone/fs/hash"
"github.com/rclone/rclone/fs/operations"
2020-06-11 14:06:02 +02:00
"github.com/rclone/rclone/lib/file"
2020-02-29 19:08:22 +01:00
"github.com/rclone/rclone/vfs/vfscommon"
)
// NB as Cache and Item are tightly linked it is necessary to have a
// total lock ordering between them. So Cache.mu must always be
// taken before Item.mu to avoid deadlocks.
//
// Cache may call into Item but care is needed if Item calls Cache
2020-04-22 13:25:40 +02:00
// FIXME need to purge cache nodes which don't have backing files and aren't dirty
// these may get created by the VFS layer or may be orphans from reload()
2020-02-29 19:08:22 +01:00
// Cache opened files
type Cache struct {
// read only - no locking needed to read these
fremote fs . Fs // fs for the remote we are caching
fcache fs . Fs // fs for the cache directory
fcacheMeta fs . Fs // fs for the cache metadata directory
opt * vfscommon . Options // vfs Options
root string // root of the cache directory
metaRoot string // root of the cache metadata directory
hashType hash . Type // hash to use locally and remotely
hashOption * fs . HashesOption // corresponding OpenOption
2020-04-17 12:18:58 +02:00
writeback * writeBack // holds Items for writeback
2020-02-29 19:08:22 +01:00
mu sync . Mutex // protects the following variables
item map [ string ] * Item // files/directories in the cache
used int64 // total size of files in the cache
}
// New creates a new cache heirachy for fremote
//
// This starts background goroutines which can be cancelled with the
// context passed in.
func New ( ctx context . Context , fremote fs . Fs , opt * vfscommon . Options ) ( * Cache , error ) {
fRoot := filepath . FromSlash ( fremote . Root ( ) )
if runtime . GOOS == "windows" {
if strings . HasPrefix ( fRoot , ` \\? ` ) {
fRoot = fRoot [ 3 : ]
}
fRoot = strings . Replace ( fRoot , ":" , "" , - 1 )
}
2020-06-11 14:06:02 +02:00
root := file . UNCPath ( filepath . Join ( config . CacheDir , "vfs" , fremote . Name ( ) , fRoot ) )
2020-02-29 19:08:22 +01:00
fs . Debugf ( nil , "vfs cache root is %q" , root )
2020-06-11 14:06:02 +02:00
metaRoot := file . UNCPath ( filepath . Join ( config . CacheDir , "vfsMeta" , fremote . Name ( ) , fRoot ) )
2020-02-29 19:08:22 +01:00
fs . Debugf ( nil , "vfs metadata cache root is %q" , root )
fcache , err := fscache . Get ( root )
if err != nil {
return nil , errors . Wrap ( err , "failed to create cache remote" )
}
fcacheMeta , err := fscache . Get ( root )
if err != nil {
return nil , errors . Wrap ( err , "failed to create cache meta remote" )
}
hashType , hashOption := operations . CommonHash ( fcache , fremote )
c := & Cache {
fremote : fremote ,
fcache : fcache ,
fcacheMeta : fcacheMeta ,
opt : opt ,
root : root ,
metaRoot : metaRoot ,
item : make ( map [ string ] * Item ) ,
hashType : hashType ,
hashOption : hashOption ,
2020-04-17 12:18:58 +02:00
writeback : newWriteBack ( ctx , opt ) ,
2020-02-29 19:08:22 +01:00
}
// Make sure cache directories exist
_ , err = c . mkdir ( "" )
if err != nil {
return nil , errors . Wrap ( err , "failed to make cache directory" )
}
// load in the cache and metadata off disk
2020-04-22 13:25:40 +02:00
err = c . reload ( ctx )
2020-02-29 19:08:22 +01:00
if err != nil {
return nil , errors . Wrap ( err , "failed to load cache" )
}
// Remove any empty directories
c . purgeEmptyDirs ( )
go c . cleaner ( ctx )
return c , nil
}
// clean returns the cleaned version of name for use in the index map
//
// name should be a remote path not an osPath
func clean ( name string ) string {
name = strings . Trim ( name , "/" )
name = path . Clean ( name )
if name == "." || name == "/" {
name = ""
}
return name
}
// toOSPath turns a remote relative name into an OS path in the cache
func ( c * Cache ) toOSPath ( name string ) string {
return filepath . Join ( c . root , filepath . FromSlash ( name ) )
}
// toOSPathMeta turns a remote relative name into an OS path in the
// cache for the metadata
func ( c * Cache ) toOSPathMeta ( name string ) string {
return filepath . Join ( c . metaRoot , filepath . FromSlash ( name ) )
}
// mkdir makes the directory for name in the cache and returns an os
// path for the file
func ( c * Cache ) mkdir ( name string ) ( string , error ) {
parent := vfscommon . FindParent ( name )
leaf := filepath . Base ( name )
parentPath := c . toOSPath ( parent )
err := os . MkdirAll ( parentPath , 0700 )
if err != nil {
return "" , errors . Wrap ( err , "make cache directory failed" )
}
parentPathMeta := c . toOSPathMeta ( parent )
err = os . MkdirAll ( parentPathMeta , 0700 )
if err != nil {
return "" , errors . Wrap ( err , "make cache meta directory failed" )
}
return filepath . Join ( parentPath , leaf ) , nil
}
// _get gets name from the cache or creates a new one
//
// It returns the item and found as to whether this item was found in
// the cache (or just created).
//
// name should be a remote path not an osPath
//
// must be called with mu held
func ( c * Cache ) _get ( name string ) ( item * Item , found bool ) {
item = c . item [ name ]
found = item != nil
if ! found {
item = newItem ( c , name )
c . item [ name ] = item
}
return item , found
}
// put puts item under name in the cache
//
// It returns an old item if there was one or nil if not.
//
// name should be a remote path not an osPath
func ( c * Cache ) put ( name string , item * Item ) ( oldItem * Item ) {
name = clean ( name )
c . mu . Lock ( )
oldItem = c . item [ name ]
if oldItem != item {
c . item [ name ] = item
} else {
oldItem = nil
}
c . mu . Unlock ( )
return oldItem
}
// InUse returns whether the name is in use in the cache
//
// name should be a remote path not an osPath
func ( c * Cache ) InUse ( name string ) bool {
name = clean ( name )
c . mu . Lock ( )
item := c . item [ name ]
c . mu . Unlock ( )
if item == nil {
return false
}
return item . inUse ( )
}
2020-04-17 12:18:58 +02:00
// DirtyItem the Item if it exists in the cache and is Dirty
//
// name should be a remote path not an osPath
func ( c * Cache ) DirtyItem ( name string ) ( item * Item ) {
name = clean ( name )
c . mu . Lock ( )
defer c . mu . Unlock ( )
item = c . item [ name ]
if item != nil && ! item . IsDirty ( ) {
item = nil
}
return item
}
2020-02-29 19:08:22 +01:00
// get gets a file name from the cache or creates a new one
//
// It returns the item and found as to whether this item was found in
// the cache (or just created).
//
// name should be a remote path not an osPath
func ( c * Cache ) get ( name string ) ( item * Item , found bool ) {
name = clean ( name )
c . mu . Lock ( )
item , found = c . _get ( name )
c . mu . Unlock ( )
return item , found
}
// Item gets a cache item for name
//
// To use it item.Open will need to be called
//
// name should be a remote path not an osPath
func ( c * Cache ) Item ( name string ) ( item * Item ) {
item , _ = c . get ( name )
return item
}
2020-04-17 12:18:58 +02:00
// Exists checks to see if the file exists in the cache or not.
2020-02-29 19:08:22 +01:00
//
2020-04-17 12:18:58 +02:00
// This is done by bringing the item into the cache which will
// validate the backing file and metadata and then asking if the Item
// exists or not.
2020-02-29 19:08:22 +01:00
func ( c * Cache ) Exists ( name string ) bool {
2020-04-17 12:18:58 +02:00
item , _ := c . get ( name )
return item . Exists ( )
2020-02-29 19:08:22 +01:00
}
// rename with os.Rename and more checking
func rename ( osOldPath , osNewPath string ) error {
sfi , err := os . Stat ( osOldPath )
if err != nil {
// Just do nothing if the source does not exist
if os . IsNotExist ( err ) {
return nil
}
return errors . Wrapf ( err , "Failed to stat source: %s" , osOldPath )
}
if ! sfi . Mode ( ) . IsRegular ( ) {
// cannot copy non-regular files (e.g., directories, symlinks, devices, etc.)
return errors . Errorf ( "Non-regular source file: %s (%q)" , sfi . Name ( ) , sfi . Mode ( ) . String ( ) )
}
dfi , err := os . Stat ( osNewPath )
if err != nil {
if ! os . IsNotExist ( err ) {
return errors . Wrapf ( err , "Failed to stat destination: %s" , osNewPath )
}
parent := vfscommon . OsFindParent ( osNewPath )
err = os . MkdirAll ( parent , 0700 )
if err != nil {
return errors . Wrapf ( err , "Failed to create parent dir: %s" , parent )
}
} else {
if ! ( dfi . Mode ( ) . IsRegular ( ) ) {
return errors . Errorf ( "Non-regular destination file: %s (%q)" , dfi . Name ( ) , dfi . Mode ( ) . String ( ) )
}
if os . SameFile ( sfi , dfi ) {
return nil
}
}
if err = os . Rename ( osOldPath , osNewPath ) ; err != nil {
return errors . Wrapf ( err , "Failed to rename in cache: %s to %s" , osOldPath , osNewPath )
}
return nil
}
// Rename the item in cache
func ( c * Cache ) Rename ( name string , newName string , newObj fs . Object ) ( err error ) {
item , _ := c . get ( name )
err = item . rename ( name , newName , newObj )
if err != nil {
return err
}
// Move the item in the cache
c . mu . Lock ( )
if item , ok := c . item [ name ] ; ok {
c . item [ newName ] = item
delete ( c . item , name )
}
c . mu . Unlock ( )
fs . Infof ( name , "Renamed in cache to %q" , newName )
return nil
}
// Remove should be called if name is deleted
2020-06-03 16:49:41 +02:00
//
// This returns true if the file was in the transfer queue so may not
// have completedly uploaded yet.
func ( c * Cache ) Remove ( name string ) ( wasWriting bool ) {
2020-04-17 12:18:58 +02:00
name = clean ( name )
c . mu . Lock ( )
item , _ := c . _get ( name )
delete ( c . item , name )
c . mu . Unlock ( )
2020-06-03 16:49:41 +02:00
return item . remove ( "file deleted" )
2020-02-29 19:08:22 +01:00
}
// SetModTime should be called to set the modification time of the cache file
func ( c * Cache ) SetModTime ( name string , modTime time . Time ) {
item , _ := c . get ( name )
item . setModTime ( modTime )
}
// CleanUp empties the cache of everything
func ( c * Cache ) CleanUp ( ) error {
err1 := os . RemoveAll ( c . root )
err2 := os . RemoveAll ( c . metaRoot )
if err1 != nil {
return err1
}
return err2
}
// walk walks the cache calling the function
func ( c * Cache ) walk ( dir string , fn func ( osPath string , fi os . FileInfo , name string ) error ) error {
return filepath . Walk ( dir , func ( osPath string , fi os . FileInfo , err error ) error {
if err != nil {
return err
}
// Find path relative to the cache root
name , err := filepath . Rel ( dir , osPath )
if err != nil {
return errors . Wrap ( err , "filepath.Rel failed in walk" )
}
if name == "." {
name = ""
}
// And convert into slashes
name = filepath . ToSlash ( name )
return fn ( osPath , fi , name )
} )
}
// reload walks the cache loading metadata files
2020-04-22 13:25:40 +02:00
//
// It iterates the files first then metadata trees. It doesn't expect
// to find any new items iterating the metadata but it will clear up
// orphan files.
func ( c * Cache ) reload ( ctx context . Context ) error {
for _ , dir := range [ ] string { c . root , c . metaRoot } {
err := c . walk ( dir , func ( osPath string , fi os . FileInfo , name string ) error {
if fi . IsDir ( ) {
return nil
}
item , found := c . get ( name )
if ! found {
err := item . reload ( ctx )
if err != nil {
fs . Errorf ( name , "vfs cache: failed to reload item: %v" , err )
}
}
return nil
} )
if err != nil {
return errors . Wrapf ( err , "failed to walk cache %q" , dir )
2020-02-29 19:08:22 +01:00
}
}
2020-04-22 13:25:40 +02:00
return nil
2020-02-29 19:08:22 +01:00
}
// purgeOld gets rid of any files that are over age
func ( c * Cache ) purgeOld ( maxAge time . Duration ) {
c . _purgeOld ( maxAge , func ( item * Item ) {
item . remove ( "too old" )
} )
}
func ( c * Cache ) _purgeOld ( maxAge time . Duration , remove func ( item * Item ) ) {
c . mu . Lock ( )
defer c . mu . Unlock ( )
cutoff := time . Now ( ) . Add ( - maxAge )
for name , item := range c . item {
if ! item . inUse ( ) {
// If not locked and access time too long ago - delete the file
dt := item . getATime ( ) . Sub ( cutoff )
// fs.Debugf(name, "atime=%v cutoff=%v, dt=%v", item.info.ATime, cutoff, dt)
if dt < 0 {
remove ( item )
// Remove the entry
delete ( c . item , name )
}
}
}
}
// Purge any empty directories
func ( c * Cache ) purgeEmptyDirs ( ) {
ctx := context . Background ( )
err := operations . Rmdirs ( ctx , c . fcache , "" , true )
if err != nil {
fs . Errorf ( c . fcache , "Failed to remove empty directories from cache: %v" , err )
}
err = operations . Rmdirs ( ctx , c . fcacheMeta , "" , true )
if err != nil {
fs . Errorf ( c . fcache , "Failed to remove empty directories from metadata cache: %v" , err )
}
}
// Remove any files that are over quota starting from the
// oldest first
func ( c * Cache ) purgeOverQuota ( quota int64 ) {
c . _purgeOverQuota ( quota , func ( item * Item ) {
item . remove ( "over quota" )
} )
}
// updateUsed updates c.used so it is accurate
func ( c * Cache ) updateUsed ( ) {
c . mu . Lock ( )
defer c . mu . Unlock ( )
newUsed := int64 ( 0 )
for _ , item := range c . item {
newUsed += item . getDiskSize ( )
}
c . used = newUsed
}
func ( c * Cache ) _purgeOverQuota ( quota int64 , remove func ( item * Item ) ) {
c . updateUsed ( )
c . mu . Lock ( )
defer c . mu . Unlock ( )
if quota <= 0 || c . used < quota {
return
}
var items Items
// Make a slice of unused files
for _ , item := range c . item {
if ! item . inUse ( ) {
items = append ( items , item )
}
}
sort . Sort ( items )
// Remove items until the quota is OK
for _ , item := range items {
if c . used < quota {
break
}
c . used -= item . getDiskSize ( )
remove ( item )
// Remove the entry
delete ( c . item , item . name )
}
}
// clean empties the cache of stuff if it can
func ( c * Cache ) clean ( ) {
// Cache may be empty so end
_ , err := os . Stat ( c . root )
if os . IsNotExist ( err ) {
return
}
c . mu . Lock ( )
oldItems , oldUsed := len ( c . item ) , fs . SizeSuffix ( c . used )
c . mu . Unlock ( )
// Remove any files that are over age
c . purgeOld ( c . opt . CacheMaxAge )
// Now remove any files that are over quota starting from the
// oldest first
c . purgeOverQuota ( int64 ( c . opt . CacheMaxSize ) )
// Stats
c . mu . Lock ( )
newItems , newUsed := len ( c . item ) , fs . SizeSuffix ( c . used )
2020-04-22 13:25:40 +02:00
totalInUse := 0
for _ , item := range c . item {
if item . inUse ( ) {
totalInUse ++
}
}
2020-02-29 19:08:22 +01:00
c . mu . Unlock ( )
2020-04-22 13:25:40 +02:00
uploadsInProgress , uploadsQueued := c . writeback . getStats ( )
2020-02-29 19:08:22 +01:00
2020-04-22 13:25:40 +02:00
fs . Infof ( nil , "Cleaned the cache: objects %d (was %d) in use %d, to upload %d, uploading %d, total size %v (was %v)" , newItems , oldItems , totalInUse , uploadsQueued , uploadsInProgress , newUsed , oldUsed )
2020-02-29 19:08:22 +01:00
}
// cleaner calls clean at regular intervals
//
// doesn't return until context is cancelled
func ( c * Cache ) cleaner ( ctx context . Context ) {
if c . opt . CachePollInterval <= 0 {
fs . Debugf ( nil , "Cache cleaning thread disabled because poll interval <= 0" )
return
}
// Start cleaning the cache immediately
c . clean ( )
// Then every interval specified
timer := time . NewTicker ( c . opt . CachePollInterval )
defer timer . Stop ( )
for {
select {
case <- timer . C :
c . clean ( )
case <- ctx . Done ( ) :
fs . Debugf ( nil , "cache cleaner exiting" )
return
}
}
}
2020-04-17 12:18:58 +02:00
// TotalInUse returns the number of items in the cache which are InUse
func ( c * Cache ) TotalInUse ( ) ( n int ) {
c . mu . Lock ( )
defer c . mu . Unlock ( )
for _ , item := range c . item {
if item . inUse ( ) {
n ++
}
}
return n
}
// Dump the cache into a string for debugging purposes
func ( c * Cache ) Dump ( ) string {
if c == nil {
return "Cache: <nil>\n"
}
c . mu . Lock ( )
defer c . mu . Unlock ( )
var out strings . Builder
out . WriteString ( "Cache{\n" )
for name , item := range c . item {
fmt . Fprintf ( & out , "\t%q: %+v,\n" , name , item )
}
out . WriteString ( "}\n" )
return out . String ( )
2020-02-29 19:08:22 +01:00
}