forked from TrueCloudLab/rclone
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:
parent
8b491f7f3d
commit
5e95877840
3 changed files with 72 additions and 4 deletions
|
@ -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 {
|
||||||
|
|
|
@ -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)
|
require.NoError(t, err)
|
||||||
|
|
||||||
file1.ModTime = t2
|
// Write some data
|
||||||
|
if write {
|
||||||
|
_, err = fd.WriteString("hello")
|
||||||
|
require.NoError(t, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
err = file.SetModTime(t2)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
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)
|
||||||
|
|
|
@ -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() {
|
||||||
|
|
Loading…
Reference in a new issue