vfs: fix modtime set if --vfs-cache-mode writes/full and no write

When using --vfs-cache-mode writes or full if a file was opened for
write intent, the modtime was set and the file was closed without
being modified the modtime would never be written back to storage.

The sequence of events

- app opens file with write intent
- app does set modtime
- rclone sets the modtime on the cache file, but not the remote file
  because it is open for write and can't be set yet
- app closes the file without changing it
- rclone doesn't upload the file because the file wasn't changed so
  the modtime doesn't get updated

This fixes the problem by making sure any unapplied modtime changes
are applied even if the file is not modified when being closed.

Fixes #4795
This commit is contained in:
Nick Craig-Wood 2021-03-14 12:17:29 +00:00
parent 8b491f7f3d
commit 5e95877840
3 changed files with 72 additions and 4 deletions

View file

@ -424,6 +424,13 @@ func (f *File) _applyPendingModTime() error {
return nil return nil
} }
// Apply a pending mod time
func (f *File) applyPendingModTime() error {
f.mu.Lock()
defer f.mu.Unlock()
return f._applyPendingModTime()
}
// _writingInProgress returns true of there are any open writers // _writingInProgress returns true of there are any open writers
// Call with read lock held // Call with read lock held
func (f *File) _writingInProgress() bool { func (f *File) _writingInProgress() bool {

View file

@ -88,17 +88,55 @@ func TestFileMethods(t *testing.T) {
assert.Equal(t, vfs, file.VFS()) assert.Equal(t, vfs, file.VFS())
} }
func TestFileSetModTime(t *testing.T) { func testFileSetModTime(t *testing.T, cacheMode vfscommon.CacheMode, open bool, write bool) {
r, vfs, file, file1, cleanup := fileCreate(t, vfscommon.CacheModeOff) if !canSetModTimeValue {
t.Skip("can't set mod time")
}
r, vfs, file, file1, cleanup := fileCreate(t, cacheMode)
defer cleanup() defer cleanup()
if !canSetModTime(t, r) { if !canSetModTime(t, r) {
t.Skip("can't set mod time") t.Skip("can't set mod time")
} }
err := file.SetModTime(t2) var (
err error
fd Handle
contents = "file1 contents"
)
if open {
// Open with write intent
if cacheMode != vfscommon.CacheModeOff {
fd, err = file.Open(os.O_WRONLY)
if write {
contents = "hello contents"
}
} else {
// Can't write without O_TRUNC with CacheMode Off
fd, err = file.Open(os.O_WRONLY | os.O_TRUNC)
if write {
contents = "hello"
} else {
contents = ""
}
}
require.NoError(t, err)
// Write some data
if write {
_, err = fd.WriteString("hello")
require.NoError(t, err)
}
}
err = file.SetModTime(t2)
require.NoError(t, err) require.NoError(t, err)
file1.ModTime = t2 if open {
require.NoError(t, fd.Close())
vfs.WaitForWriters(waitForWritersDelay)
}
file1 = fstest.NewItem(file1.Path, contents, t2)
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
vfs.Opt.ReadOnly = true vfs.Opt.ReadOnly = true
@ -106,6 +144,26 @@ func TestFileSetModTime(t *testing.T) {
assert.Equal(t, EROFS, err) assert.Equal(t, EROFS, err)
} }
// Test various combinations of setting mod times with and
// without the cache and with and without opening or writing
// to the file.
//
// Each of these tests a different path through the VFS code.
func TestFileSetModTime(t *testing.T) {
for _, cacheMode := range []vfscommon.CacheMode{vfscommon.CacheModeOff, vfscommon.CacheModeFull} {
for _, open := range []bool{false, true} {
for _, write := range []bool{false, true} {
if write && !open {
continue
}
t.Run(fmt.Sprintf("cache=%v,open=%v,write=%v", cacheMode, open, write), func(t *testing.T) {
testFileSetModTime(t, cacheMode, open, write)
})
}
}
}
}
func fileCheckContents(t *testing.T, file *File) { func fileCheckContents(t *testing.T, file *File) {
fd, err := file.Open(os.O_RDONLY) fd, err := file.Open(os.O_RDONLY)
require.NoError(t, err) require.NoError(t, err)

View file

@ -158,6 +158,9 @@ func (fh *RWFileHandle) close() (err error) {
if fh.opened { if fh.opened {
err = fh.item.Close(fh.file.setObject) err = fh.item.Close(fh.file.setObject)
fh.opened = false fh.opened = false
} else {
// apply any pending mod times if any
_ = fh.file.applyPendingModTime()
} }
if !fh.readOnly() { if !fh.readOnly() {