From b8c2544dcb9559ac30d9becdfb5fe7b0444a44eb Mon Sep 17 00:00:00 2001 From: Courtney Bane Date: Tue, 19 Mar 2019 20:27:37 -0500 Subject: [PATCH 1/4] Examine file ctime when checking if files have changed. --- changelog/unreleased/issue-2179 | 8 ++++++++ internal/archiver/archiver.go | 7 ++++++- internal/archiver/archiver_test.go | 21 +++++++++++++++++++++ internal/fs/stat.go | 1 + internal/fs/stat_bsd.go | 1 + internal/fs/stat_unix.go | 1 + internal/fs/stat_windows.go | 2 ++ 7 files changed, 40 insertions(+), 1 deletion(-) create mode 100644 changelog/unreleased/issue-2179 diff --git a/changelog/unreleased/issue-2179 b/changelog/unreleased/issue-2179 new file mode 100644 index 000000000..c0a275b04 --- /dev/null +++ b/changelog/unreleased/issue-2179 @@ -0,0 +1,8 @@ +Bugfix: Use ctime when checking for file changes + +Previously, restic only checked a file's mtime (along with other non-timestamp +data) to decide if a file has changed. This could cause it to not notice changes +if something edits a file and then resets the timestamp. Restic now also checks +the ctime, so any modification to a file should be noticed. + +https://github.com/restic/restic/issues/2179 diff --git a/internal/archiver/archiver.go b/internal/archiver/archiver.go index b21f79e87..e612ab8e7 100644 --- a/internal/archiver/archiver.go +++ b/internal/archiver/archiver.go @@ -453,8 +453,13 @@ func fileChanged(fi os.FileInfo, node *restic.Node, ignoreInode bool) bool { return true } - // check size + // check status change timestamp extFI := fs.ExtendedStat(fi) + if !extFI.ChangeTime.Equal(node.ChangeTime) { + return true + } + + // check size if uint64(fi.Size()) != node.Size || uint64(extFI.Size) != node.Size { return true } diff --git a/internal/archiver/archiver_test.go b/internal/archiver/archiver_test.go index 419b3a91c..ba22b8abe 100644 --- a/internal/archiver/archiver_test.go +++ b/internal/archiver/archiver_test.go @@ -1,6 +1,7 @@ package archiver import ( + "bytes" "context" "io/ioutil" "os" @@ -576,6 +577,26 @@ func TestFileChanged(t *testing.T) { save(t, filename, defaultContent) }, }, + { + Name: "new-content-same-timestamp", + Modify: func(t testing.TB, filename string) { + fi, _ := os.Stat(filename) + extFI := fs.ExtendedStat(fi) + save(t, filename, bytes.ToUpper(defaultContent)) + sleep() + ts := []syscall.Timespec{ + { + Sec: extFI.AccessTime.Unix(), + Nsec: int64(extFI.AccessTime.Nanosecond()), + }, + { + Sec: extFI.ModTime.Unix(), + Nsec: int64(extFI.ModTime.Nanosecond()), + }, + } + syscall.UtimesNano(filename, ts) + }, + }, { Name: "other-content", Modify: func(t testing.TB, filename string) { diff --git a/internal/fs/stat.go b/internal/fs/stat.go index d37d12942..e1006fd61 100644 --- a/internal/fs/stat.go +++ b/internal/fs/stat.go @@ -22,6 +22,7 @@ type ExtendedFileInfo struct { AccessTime time.Time // last access time stamp ModTime time.Time // last (content) modification time stamp + ChangeTime time.Time // last status change time stamp } // ExtendedStat returns an ExtendedFileInfo constructed from the os.FileInfo. diff --git a/internal/fs/stat_bsd.go b/internal/fs/stat_bsd.go index 62a258e64..d5e8ce550 100644 --- a/internal/fs/stat_bsd.go +++ b/internal/fs/stat_bsd.go @@ -30,6 +30,7 @@ func extendedStat(fi os.FileInfo) ExtendedFileInfo { AccessTime: time.Unix(s.Atimespec.Unix()), ModTime: time.Unix(s.Mtimespec.Unix()), + ChangeTime: time.Unix(s.Ctimespec.Unix()), } return extFI diff --git a/internal/fs/stat_unix.go b/internal/fs/stat_unix.go index 56c22f8bc..34b98a31e 100644 --- a/internal/fs/stat_unix.go +++ b/internal/fs/stat_unix.go @@ -30,6 +30,7 @@ func extendedStat(fi os.FileInfo) ExtendedFileInfo { AccessTime: time.Unix(s.Atim.Unix()), ModTime: time.Unix(s.Mtim.Unix()), + ChangeTime: time.Unix(s.Ctim.Unix()), } return extFI diff --git a/internal/fs/stat_windows.go b/internal/fs/stat_windows.go index 16f9fe0eb..a8f13ccea 100644 --- a/internal/fs/stat_windows.go +++ b/internal/fs/stat_windows.go @@ -27,5 +27,7 @@ func extendedStat(fi os.FileInfo) ExtendedFileInfo { mtime := syscall.NsecToTimespec(s.LastWriteTime.Nanoseconds()) extFI.ModTime = time.Unix(mtime.Unix()) + extFI.ChangeTime = extFI.ModTime + return extFI } From 35b76078026f94e194e745fac394b332a47b8d99 Mon Sep 17 00:00:00 2001 From: Courtney Bane Date: Tue, 23 Apr 2019 22:27:38 -0500 Subject: [PATCH 2/4] Don't check ctime when ignoring inode. --- internal/archiver/archiver.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/internal/archiver/archiver.go b/internal/archiver/archiver.go index e612ab8e7..34e6df6c2 100644 --- a/internal/archiver/archiver.go +++ b/internal/archiver/archiver.go @@ -455,7 +455,7 @@ func fileChanged(fi os.FileInfo, node *restic.Node, ignoreInode bool) bool { // check status change timestamp extFI := fs.ExtendedStat(fi) - if !extFI.ChangeTime.Equal(node.ChangeTime) { + if !ignoreInode && !extFI.ChangeTime.Equal(node.ChangeTime) { return true } From 0ebfc55ee37bbc7c2f58a6af4a78de60a776eb99 Mon Sep 17 00:00:00 2001 From: Courtney Bane Date: Tue, 23 Apr 2019 22:39:13 -0500 Subject: [PATCH 3/4] Use existing setTimestamp function for ctime test and improve error checking. --- internal/archiver/archiver_test.go | 17 +++++------------ 1 file changed, 5 insertions(+), 12 deletions(-) diff --git a/internal/archiver/archiver_test.go b/internal/archiver/archiver_test.go index ba22b8abe..f15de610b 100644 --- a/internal/archiver/archiver_test.go +++ b/internal/archiver/archiver_test.go @@ -580,21 +580,14 @@ func TestFileChanged(t *testing.T) { { Name: "new-content-same-timestamp", Modify: func(t testing.TB, filename string) { - fi, _ := os.Stat(filename) + fi, err := os.Stat(filename) + if err != nil { + t.Fatal(err) + } extFI := fs.ExtendedStat(fi) save(t, filename, bytes.ToUpper(defaultContent)) sleep() - ts := []syscall.Timespec{ - { - Sec: extFI.AccessTime.Unix(), - Nsec: int64(extFI.AccessTime.Nanosecond()), - }, - { - Sec: extFI.ModTime.Unix(), - Nsec: int64(extFI.ModTime.Nanosecond()), - }, - } - syscall.UtimesNano(filename, ts) + setTimestamp(t, filename, extFI.AccessTime, extFI.ModTime) }, }, { From 00b527fb093426ca9636932690cd188824728248 Mon Sep 17 00:00:00 2001 From: Courtney Bane Date: Wed, 24 Apr 2019 20:54:15 -0500 Subject: [PATCH 4/4] Update changelog text, and add pull request link. --- changelog/unreleased/issue-2179 | 15 +++++++++++---- 1 file changed, 11 insertions(+), 4 deletions(-) diff --git a/changelog/unreleased/issue-2179 b/changelog/unreleased/issue-2179 index c0a275b04..e87778d17 100644 --- a/changelog/unreleased/issue-2179 +++ b/changelog/unreleased/issue-2179 @@ -1,8 +1,15 @@ -Bugfix: Use ctime when checking for file changes +Enhancement: Use ctime when checking for file changes Previously, restic only checked a file's mtime (along with other non-timestamp -data) to decide if a file has changed. This could cause it to not notice changes -if something edits a file and then resets the timestamp. Restic now also checks -the ctime, so any modification to a file should be noticed. +metadata) to decide if a file has changed. This could cause restic to not notice +that a file has changed (and therefore continue to store the old version, as +opposed to the modified version) if something edits the file and then resets the +timestamp. Restic now also checks the ctime of files, so any modifications to a +file should be noticed, and the modified file will be backed up. The ctime check +will be disabled if the --ignore-inode flag was given. + +If this change causes problems for you, please open an issue, and we can look in +to adding a seperate flag to disable just the ctime check. https://github.com/restic/restic/issues/2179 +https://github.com/restic/restic/pull/2212