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:
parent
c72d2c67ed
commit
4cc2a7f342
3 changed files with 21 additions and 4 deletions
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in a new issue