mount: fix caching of old directories after renaming them

It was discovered `rclone mount` (but not `rclone cmount`) cached
directories after rename which it shouldn't have done.

This caused IO errors when trying to access files in renamed
directories on bucket based file systems.

This turned out to be the kernel caching the directories as basil/fuse
sets their expiry time to 60s for some reason.

This fix invalidates the relevant kernel cache entries in the for the
directories which fixes the problem.

Fixes: #4977
See: https://forum.rclone.org/t/after-a-directory-renmane-using-mv-files-are-not-visible-any-longer/22797
This commit is contained in:
Nick Craig-Wood 2021-03-16 13:23:36 +00:00
parent c72d2c67ed
commit 4cc2a7f342
3 changed files with 21 additions and 4 deletions

View file

@ -181,6 +181,15 @@ func (d *Dir) Remove(ctx context.Context, req *fuse.RemoveRequest) (err error) {
return nil return nil
} }
// Invalidate a leaf in a directory
func (d *Dir) invalidateEntry(dirNode fusefs.Node, leaf string) {
fs.Debugf(dirNode, "Invalidating %q", leaf)
err := d.fsys.server.InvalidateEntry(dirNode, leaf)
if err != nil {
fs.Debugf(dirNode, "Failed to invalidate %q: %v", leaf, err)
}
}
// Check interface satisfied // Check interface satisfied
var _ fusefs.NodeRenamer = (*Dir)(nil) var _ fusefs.NodeRenamer = (*Dir)(nil)
@ -197,6 +206,13 @@ func (d *Dir) Rename(ctx context.Context, req *fuse.RenameRequest, newDir fusefs
return translateError(err) return translateError(err)
} }
// Invalidate the new directory entry so it gets re-read (in
// the background otherwise we cause a deadlock)
//
// See https://github.com/rclone/rclone/issues/4977 for why
go d.invalidateEntry(newDir, req.NewName)
//go d.invalidateEntry(d, req.OldName)
return nil return nil
} }

View file

@ -22,6 +22,7 @@ type FS struct {
*vfs.VFS *vfs.VFS
f fs.Fs f fs.Fs
opt *mountlib.Options opt *mountlib.Options
server *fusefs.Server
} }
// Check interface satisfied // Check interface satisfied

View file

@ -91,12 +91,12 @@ func mount(VFS *vfs.VFS, mountpoint string, opt *mountlib.Options) (<-chan error
} }
filesys := NewFS(VFS, opt) filesys := NewFS(VFS, opt)
server := fusefs.New(c, nil) filesys.server = fusefs.New(c, nil)
// Serve the mount point in the background returning error to errChan // Serve the mount point in the background returning error to errChan
errChan := make(chan error, 1) errChan := make(chan error, 1)
go func() { go func() {
err := server.Serve(filesys) err := filesys.server.Serve(filesys)
closeErr := c.Close() closeErr := c.Close()
if err == nil { if err == nil {
err = closeErr err = closeErr