lib/model: Handle deleted-then-ignored files (fixes #3502)

When files that were previously marked as deleted became ignored, we
used to do nothing at all. This changes that behavior to set the Invalid
bit (that we should rename to Ignored). This then becomes an update to
other devices that they should not trust our knowledge about the file in
question.

Read this diff without whitespace...

Tested by
- creating a bunch of files on s1
- letting them sync to s2
- shutting down s2
- deleting the files on s1 and rescanning
- adding the files to .stignore on s1 and rescanning
- starting up s2 and letting it sync
- observing the files are not deleted on s2, and it considers itself up
  to date.

GitHub-Pull-Request: https://github.com/syncthing/syncthing/pull/3557
This commit is contained in:
Jakob Borg 2016-09-02 13:23:24 +00:00 committed by Audrius Butkevicius
parent 69b7f26e4c
commit 46a143e80e

View File

@ -1735,41 +1735,46 @@ func (m *Model) internalScanFolderSubdirs(folder string, subDirs []string) error
subDirs = []string{""} subDirs = []string{""}
} }
// Do a scan of the database for each prefix, to check for deleted files. // Do a scan of the database for each prefix, to check for deleted and
// ignored files.
batch = batch[:0] batch = batch[:0]
for _, sub := range subDirs { for _, sub := range subDirs {
var iterError error var iterError error
fs.WithPrefixedHaveTruncated(protocol.LocalDeviceID, sub, func(fi db.FileIntf) bool { fs.WithPrefixedHaveTruncated(protocol.LocalDeviceID, sub, func(fi db.FileIntf) bool {
f := fi.(db.FileInfoTruncated) f := fi.(db.FileInfoTruncated)
if !f.IsDeleted() { if len(batch) == batchSizeFiles {
if len(batch) == batchSizeFiles { if err := m.CheckFolderHealth(folder); err != nil {
if err := m.CheckFolderHealth(folder); err != nil { iterError = err
iterError = err return false
return false
}
m.updateLocalsFromScanning(folder, batch)
batch = batch[:0]
} }
m.updateLocalsFromScanning(folder, batch)
batch = batch[:0]
}
if !f.IsInvalid() && (ignores.Match(f.Name).IsIgnored() || symlinkInvalid(folder, f)) { switch {
// File has been ignored or an unsupported symlink. Set invalid bit. case !f.IsInvalid() && (ignores.Match(f.Name).IsIgnored() || symlinkInvalid(folder, f)):
l.Debugln("setting invalid bit on ignored", f) // File was valid at last pass but has been ignored or is an
nf := protocol.FileInfo{ // unsupported symlink. Set invalid bit.
Name: f.Name, l.Debugln("setting invalid bit on ignored", f)
Type: f.Type, nf := protocol.FileInfo{
Size: f.Size, Name: f.Name,
ModifiedS: f.ModifiedS, Type: f.Type,
ModifiedNs: f.ModifiedNs, Size: f.Size,
Permissions: f.Permissions, ModifiedS: f.ModifiedS,
NoPermissions: f.NoPermissions, ModifiedNs: f.ModifiedNs,
Invalid: true, Permissions: f.Permissions,
Version: f.Version, // The file is still the same, so don't bump version NoPermissions: f.NoPermissions,
} Invalid: true,
batch = append(batch, nf) Version: f.Version, // The file is still the same, so don't bump version
} else if _, err := mtimefs.Lstat(filepath.Join(folderCfg.Path(), f.Name)); err != nil { }
// File has been deleted. batch = append(batch, nf)
case !f.IsInvalid() && !f.IsDeleted():
// The file is valid and not deleted. Lets check if it's
// still here.
if _, err := mtimefs.Lstat(filepath.Join(folderCfg.Path(), f.Name)); err != nil {
// We don't specifically verify that the error is // We don't specifically verify that the error is
// os.IsNotExist because there is a corner case when a // os.IsNotExist because there is a corner case when a
// directory is suddenly transformed into a file. When that // directory is suddenly transformed into a file. When that