vfs: fix rename of open files when using the VFS cache

Before this change, renaming an open file when using the VFS cache was
delayed until the file was closed.  This meant that the file was not
readable after a rename even though it is was in the cache.

After this change we rename the local cache file and the in memory
cache, delaying only the rename of the file in object storage.

See: https://forum.rclone.org/t/xen-orchestra-ebadf-bad-file-descriptor-write/13104
This commit is contained in:
Nick Craig-Wood 2019-12-05 12:58:24 +00:00
parent 241921c786
commit 6e683b4359
3 changed files with 146 additions and 32 deletions

View file

@ -137,29 +137,42 @@ func (f *File) applyPendingRename() {
func (f *File) rename(ctx context.Context, destDir *Dir, newName string) error { func (f *File) rename(ctx context.Context, destDir *Dir, newName string) error {
f.mu.RLock() f.mu.RLock()
d := f.d d := f.d
o := f.o
oldPendingRenameFun := f.pendingRenameFun
f.mu.RUnlock() f.mu.RUnlock()
if features := d.f.Features(); features.Move == nil && features.Copy == nil { if features := d.f.Features(); features.Move == nil && features.Copy == nil {
err := errors.Errorf("Fs %q can't rename files (no server side Move or Copy)", d.f) err := errors.Errorf("Fs %q can't rename files (no server side Move or Copy)", d.f)
fs.Errorf(f.Path(), "Dir.Rename error: %v", err) fs.Errorf(f.Path(), "Dir.Rename error: %v", err)
return err return err
} }
newPath := path.Join(destDir.path, newName)
renameCall := func(ctx context.Context) error { renameCall := func(ctx context.Context) error {
newPath := path.Join(destDir.path, newName) f.mu.RLock()
o := f.o
f.mu.RUnlock()
if o == nil {
return errors.New("Cannot rename: file object is not available")
}
// chain rename calls if any
if oldPendingRenameFun != nil {
err := oldPendingRenameFun(ctx)
if err != nil {
return err
}
}
// do the move of the remote object
dstOverwritten, _ := d.f.NewObject(ctx, newPath) dstOverwritten, _ := d.f.NewObject(ctx, newPath)
newObject, err := operations.Move(ctx, d.f, dstOverwritten, newPath, f.o) newObject, err := operations.Move(ctx, d.f, dstOverwritten, newPath, o)
if err != nil { if err != nil {
fs.Errorf(f.Path(), "File.Rename error: %v", err) fs.Errorf(f.Path(), "File.Rename error: %v", err)
return err return err
} }
// Rename in the cache too if it exists
if f.d.vfs.Opt.CacheMode >= CacheModeWrites && f.d.vfs.cache.exists(f.Path()) {
if err := f.d.vfs.cache.rename(f.Path(), newPath); err != nil {
fs.Infof(f.Path(), "File.Rename failed in Cache: %v", err)
}
}
// newObject can be nil here for example if --dry-run // newObject can be nil here for example if --dry-run
if newObject == nil { if newObject == nil {
err = errors.New("rename failed: nil object returned") err = errors.New("rename failed: nil object returned")
@ -167,25 +180,32 @@ func (f *File) rename(ctx context.Context, destDir *Dir, newName string) error {
return err return err
} }
// Update the node with the new details // Update the node with the new details
fs.Debugf(f.o, "Updating file with %v %p", newObject, f) fs.Debugf(o, "Updating file with %v %p", newObject, f)
// f.rename(destDir, newObject) // f.rename(destDir, newObject)
f.mu.Lock() f.mu.Lock()
f.o = newObject f.o = newObject
f.d = destDir
f.leaf = path.Base(newObject.Remote())
f.pendingRenameFun = nil f.pendingRenameFun = nil
f.mu.Unlock() f.mu.Unlock()
return nil return nil
} }
f.mu.RLock() // Rename in the cache if it exists
if f.d.vfs.Opt.CacheMode != CacheModeOff && f.d.vfs.cache.exists(f.Path()) {
if err := f.d.vfs.cache.rename(f.Path(), newPath); err != nil {
fs.Infof(f.Path(), "File.Rename failed in Cache: %v", err)
}
}
// rename the file object
f.mu.Lock()
f.d = destDir
f.leaf = newName
writing := f._writingInProgress() writing := f._writingInProgress()
f.mu.RUnlock() f.mu.Unlock()
if writing { if writing {
fs.Debugf(f.o, "File is currently open, delaying rename %p", f) fs.Debugf(o, "File is currently open, delaying rename %p", f)
f.mu.Lock() f.mu.Lock()
f.d = destDir
f.leaf = newName
f.pendingRenameFun = renameCall f.pendingRenameFun = renameCall
f.mu.Unlock() f.mu.Unlock()
return nil return nil

View file

@ -6,6 +6,7 @@ import (
"os" "os"
"testing" "testing"
"github.com/rclone/rclone/fs"
"github.com/rclone/rclone/fstest" "github.com/rclone/rclone/fstest"
"github.com/rclone/rclone/fstest/mockfs" "github.com/rclone/rclone/fstest/mockfs"
"github.com/rclone/rclone/fstest/mockobject" "github.com/rclone/rclone/fstest/mockobject"
@ -13,8 +14,10 @@ import (
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
) )
func fileCreate(t *testing.T, r *fstest.Run) (*VFS, *File, fstest.Item) { func fileCreate(t *testing.T, r *fstest.Run, mode CacheMode) (*VFS, *File, fstest.Item) {
vfs := New(r.Fremote, nil) opt := DefaultOpt
opt.CacheMode = mode
vfs := New(r.Fremote, &opt)
file1 := r.WriteObject(context.Background(), "dir/file1", "file1 contents", t1) file1 := r.WriteObject(context.Background(), "dir/file1", "file1 contents", t1)
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
@ -29,7 +32,7 @@ func fileCreate(t *testing.T, r *fstest.Run) (*VFS, *File, fstest.Item) {
func TestFileMethods(t *testing.T) { func TestFileMethods(t *testing.T) {
r := fstest.NewRun(t) r := fstest.NewRun(t)
defer r.Finalise() defer r.Finalise()
vfs, file, _ := fileCreate(t, r) vfs, file, _ := fileCreate(t, r, CacheModeOff)
// String // String
assert.Equal(t, "dir/file1", file.String()) assert.Equal(t, "dir/file1", file.String())
@ -84,7 +87,7 @@ func TestFileSetModTime(t *testing.T) {
return return
} }
defer r.Finalise() defer r.Finalise()
vfs, file, file1 := fileCreate(t, r) vfs, file, file1 := fileCreate(t, r, CacheModeOff)
err := file.SetModTime(t2) err := file.SetModTime(t2)
require.NoError(t, err) require.NoError(t, err)
@ -97,12 +100,8 @@ func TestFileSetModTime(t *testing.T) {
assert.Equal(t, EROFS, err) assert.Equal(t, EROFS, err)
} }
func TestFileOpenRead(t *testing.T) { func fileCheckContents(t *testing.T, file *File) {
r := fstest.NewRun(t) fd, err := file.Open(os.O_RDONLY)
defer r.Finalise()
_, file, _ := fileCreate(t, r)
fd, err := file.openRead()
require.NoError(t, err) require.NoError(t, err)
contents, err := ioutil.ReadAll(fd) contents, err := ioutil.ReadAll(fd)
@ -112,6 +111,14 @@ func TestFileOpenRead(t *testing.T) {
require.NoError(t, fd.Close()) require.NoError(t, fd.Close())
} }
func TestFileOpenRead(t *testing.T) {
r := fstest.NewRun(t)
defer r.Finalise()
_, file, _ := fileCreate(t, r, CacheModeOff)
fileCheckContents(t, file)
}
func TestFileOpenReadUnknownSize(t *testing.T) { func TestFileOpenReadUnknownSize(t *testing.T) {
var ( var (
contents = []byte("file contents") contents = []byte("file contents")
@ -160,7 +167,7 @@ func TestFileOpenReadUnknownSize(t *testing.T) {
func TestFileOpenWrite(t *testing.T) { func TestFileOpenWrite(t *testing.T) {
r := fstest.NewRun(t) r := fstest.NewRun(t)
defer r.Finalise() defer r.Finalise()
vfs, file, _ := fileCreate(t, r) vfs, file, _ := fileCreate(t, r, CacheModeOff)
fd, err := file.openWrite(os.O_WRONLY | os.O_TRUNC) fd, err := file.openWrite(os.O_WRONLY | os.O_TRUNC)
require.NoError(t, err) require.NoError(t, err)
@ -181,7 +188,7 @@ func TestFileOpenWrite(t *testing.T) {
func TestFileRemove(t *testing.T) { func TestFileRemove(t *testing.T) {
r := fstest.NewRun(t) r := fstest.NewRun(t)
defer r.Finalise() defer r.Finalise()
vfs, file, _ := fileCreate(t, r) vfs, file, _ := fileCreate(t, r, CacheModeOff)
err := file.Remove() err := file.Remove()
require.NoError(t, err) require.NoError(t, err)
@ -196,7 +203,7 @@ func TestFileRemove(t *testing.T) {
func TestFileRemoveAll(t *testing.T) { func TestFileRemoveAll(t *testing.T) {
r := fstest.NewRun(t) r := fstest.NewRun(t)
defer r.Finalise() defer r.Finalise()
vfs, file, _ := fileCreate(t, r) vfs, file, _ := fileCreate(t, r, CacheModeOff)
err := file.RemoveAll() err := file.RemoveAll()
require.NoError(t, err) require.NoError(t, err)
@ -211,7 +218,7 @@ func TestFileRemoveAll(t *testing.T) {
func TestFileOpen(t *testing.T) { func TestFileOpen(t *testing.T) {
r := fstest.NewRun(t) r := fstest.NewRun(t)
defer r.Finalise() defer r.Finalise()
_, file, _ := fileCreate(t, r) _, file, _ := fileCreate(t, r, CacheModeOff)
fd, err := file.Open(os.O_RDONLY) fd, err := file.Open(os.O_RDONLY)
require.NoError(t, err) require.NoError(t, err)
@ -233,3 +240,90 @@ func TestFileOpen(t *testing.T) {
fd, err = file.Open(3) fd, err = file.Open(3)
assert.Equal(t, EPERM, err) assert.Equal(t, EPERM, err)
} }
func testFileRename(t *testing.T, mode CacheMode) {
r := fstest.NewRun(t)
defer r.Finalise()
vfs, file, item := fileCreate(t, r, mode)
rootDir, err := vfs.Root()
require.NoError(t, err)
// check file in cache
if mode != CacheModeOff {
// read contents to get file in cache
fileCheckContents(t, file)
assert.True(t, vfs.cache.exists(item.Path))
}
dir := file.Dir()
// start with "dir/file1"
fstest.CheckItems(t, r.Fremote, item)
// rename file to "newLeaf"
err = dir.Rename("file1", "newLeaf", rootDir)
require.NoError(t, err)
item.Path = "newLeaf"
fstest.CheckItems(t, r.Fremote, item)
// check file in cache
if mode != CacheModeOff {
assert.True(t, vfs.cache.exists(item.Path))
}
// check file exists in the vfs layer at its new name
_, err = vfs.Stat("newLeaf")
require.NoError(t, err)
// rename it back to "dir/file1"
err = rootDir.Rename("newLeaf", "file1", dir)
require.NoError(t, err)
item.Path = "dir/file1"
fstest.CheckItems(t, r.Fremote, item)
// check file in cache
if mode != CacheModeOff {
assert.True(t, vfs.cache.exists(item.Path))
}
// now try renaming it with the file open
// first open it and write to it but dont close it
fd, err := file.Open(os.O_WRONLY | os.O_TRUNC)
require.NoError(t, err)
newContents := []byte("this is some new contents")
_, err = fd.Write(newContents)
require.NoError(t, err)
// rename file to "newLeaf"
err = dir.Rename("file1", "newLeaf", rootDir)
require.NoError(t, err)
newItem := fstest.NewItem("newLeaf", string(newContents), item.ModTime)
// check file has been renamed immediately in the cache
if mode != CacheModeOff {
assert.True(t, vfs.cache.exists("newLeaf"))
}
// check file exists in the vfs layer at its new name
_, err = vfs.Stat("newLeaf")
require.NoError(t, err)
// Close the file
require.NoError(t, fd.Close())
// Check file has now been renamed on the remote
item.Path = "newLeaf"
fstest.CheckListingWithPrecision(t, r.Fremote, []fstest.Item{newItem}, nil, fs.ModTimeNotSupported)
}
func TestFileRename(t *testing.T) {
t.Run("CacheModeOff", func(t *testing.T) {
testFileRename(t, CacheModeOff)
})
t.Run("CacheModeFull", func(t *testing.T) {
testFileRename(t, CacheModeFull)
})
}

View file

@ -698,7 +698,7 @@ func TestRWFileModTimeWithOpenWriters(t *testing.T) {
} }
} }
func TestCacheRename(t *testing.T) { func TestRWCacheRename(t *testing.T) {
r := fstest.NewRun(t) r := fstest.NewRun(t)
defer r.Finalise() defer r.Finalise()