| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265 | 
							- // Copyright (C) 2016 The Syncthing Authors.
 
- //
 
- // This Source Code Form is subject to the terms of the Mozilla Public
 
- // License, v. 2.0. If a copy of the MPL was not distributed with this file,
 
- // You can obtain one at https://mozilla.org/MPL/2.0/.
 
- package fs
 
- import (
 
- 	"errors"
 
- 	"os"
 
- 	"path/filepath"
 
- 	"testing"
 
- 	"time"
 
- 	"github.com/syncthing/syncthing/lib/build"
 
- )
 
- func TestMtimeFS(t *testing.T) {
 
- 	td := t.TempDir()
 
- 	os.Mkdir(filepath.Join(td, "testdata"), 0o755)
 
- 	os.WriteFile(filepath.Join(td, "testdata", "exists0"), []byte("hello"), 0o644)
 
- 	os.WriteFile(filepath.Join(td, "testdata", "exists1"), []byte("hello"), 0o644)
 
- 	os.WriteFile(filepath.Join(td, "testdata", "exists2"), []byte("hello"), 0o644)
 
- 	// a random time with nanosecond precision
 
- 	testTime := time.Unix(1234567890, 123456789)
 
- 	mtimefs := newMtimeFS(td, make(mapStore))
 
- 	// Do one Chtimes call that will go through to the normal filesystem
 
- 	mtimefs.chtimes = os.Chtimes
 
- 	if err := mtimefs.Chtimes("testdata/exists0", testTime, testTime); err != nil {
 
- 		t.Error("Should not have failed:", err)
 
- 	}
 
- 	// Do one call that gets an error back from the underlying Chtimes
 
- 	mtimefs.chtimes = failChtimes
 
- 	if err := mtimefs.Chtimes("testdata/exists1", testTime, testTime); err != nil {
 
- 		t.Error("Should not have failed:", err)
 
- 	}
 
- 	// Do one call that gets struck by an exceptionally evil Chtimes
 
- 	mtimefs.chtimes = evilChtimes
 
- 	if err := mtimefs.Chtimes("testdata/exists2", testTime, testTime); err != nil {
 
- 		t.Error("Should not have failed:", err)
 
- 	}
 
- 	// All of the calls were successful, so an Lstat on them should return
 
- 	// the test timestamp.
 
- 	for _, file := range []string{"testdata/exists0", "testdata/exists1", "testdata/exists2"} {
 
- 		if info, err := mtimefs.Lstat(file); err != nil {
 
- 			t.Error("Lstat shouldn't fail:", err)
 
- 		} else if !info.ModTime().Equal(testTime) {
 
- 			t.Errorf("Time mismatch; %v != expected %v", info.ModTime(), testTime)
 
- 		}
 
- 	}
 
- 	// The two last files should certainly not have the correct timestamp
 
- 	// when looking directly on disk though.
 
- 	for _, file := range []string{"testdata/exists1", "testdata/exists2"} {
 
- 		if info, err := os.Lstat(filepath.Join(td, file)); err != nil {
 
- 			t.Error("Lstat shouldn't fail:", err)
 
- 		} else if info.ModTime().Equal(testTime) {
 
- 			t.Errorf("Unexpected time match; %v == %v", info.ModTime(), testTime)
 
- 		}
 
- 	}
 
- 	// Changing the timestamp on disk should be reflected in a new Lstat
 
- 	// call. Choose a time that is likely to be able to be on all reasonable
 
- 	// filesystems.
 
- 	testTime = time.Now().Add(5 * time.Hour).Truncate(time.Minute)
 
- 	os.Chtimes(filepath.Join(td, "testdata/exists0"), testTime, testTime)
 
- 	if info, err := mtimefs.Lstat("testdata/exists0"); err != nil {
 
- 		t.Error("Lstat shouldn't fail:", err)
 
- 	} else if !info.ModTime().Equal(testTime) {
 
- 		t.Errorf("Time mismatch; %v != expected %v", info.ModTime(), testTime)
 
- 	}
 
- }
 
- func TestMtimeFSWalk(t *testing.T) {
 
- 	dir := t.TempDir()
 
- 	mtimefs, walkFs := newMtimeFSWithWalk(dir, make(mapStore))
 
- 	underlying := mtimefs.Filesystem
 
- 	mtimefs.chtimes = failChtimes
 
- 	if err := os.WriteFile(filepath.Join(dir, "file"), []byte("hello"), 0o644); err != nil {
 
- 		t.Fatal(err)
 
- 	}
 
- 	oldStat, err := mtimefs.Lstat("file")
 
- 	if err != nil {
 
- 		t.Fatal(err)
 
- 	}
 
- 	newTime := time.Now().Add(-2 * time.Hour)
 
- 	if err := mtimefs.Chtimes("file", newTime, newTime); err != nil {
 
- 		t.Fatal(err)
 
- 	}
 
- 	if newStat, err := mtimefs.Lstat("file"); err != nil {
 
- 		t.Fatal(err)
 
- 	} else if !newStat.ModTime().Equal(newTime) {
 
- 		t.Errorf("expected time %v, lstat time %v", newTime, newStat.ModTime())
 
- 	}
 
- 	if underlyingStat, err := underlying.Lstat("file"); err != nil {
 
- 		t.Fatal(err)
 
- 	} else if !underlyingStat.ModTime().Equal(oldStat.ModTime()) {
 
- 		t.Errorf("expected time %v, lstat time %v", oldStat.ModTime(), underlyingStat.ModTime())
 
- 	}
 
- 	found := false
 
- 	_ = walkFs.Walk("", func(path string, info FileInfo, err error) error {
 
- 		if path == "file" {
 
- 			found = true
 
- 			if !info.ModTime().Equal(newTime) {
 
- 				t.Errorf("expected time %v, lstat time %v", newTime, info.ModTime())
 
- 			}
 
- 		}
 
- 		return nil
 
- 	})
 
- 	if !found {
 
- 		t.Error("did not find")
 
- 	}
 
- }
 
- func TestMtimeFSOpen(t *testing.T) {
 
- 	dir := t.TempDir()
 
- 	mtimefs := newMtimeFS(dir, make(mapStore))
 
- 	underlying := mtimefs.Filesystem
 
- 	mtimefs.chtimes = failChtimes
 
- 	if err := os.WriteFile(filepath.Join(dir, "file"), []byte("hello"), 0o644); err != nil {
 
- 		t.Fatal(err)
 
- 	}
 
- 	oldStat, err := mtimefs.Lstat("file")
 
- 	if err != nil {
 
- 		t.Fatal(err)
 
- 	}
 
- 	newTime := time.Now().Add(-2 * time.Hour)
 
- 	if err := mtimefs.Chtimes("file", newTime, newTime); err != nil {
 
- 		t.Fatal(err)
 
- 	}
 
- 	if newStat, err := mtimefs.Lstat("file"); err != nil {
 
- 		t.Fatal(err)
 
- 	} else if !newStat.ModTime().Equal(newTime) {
 
- 		t.Errorf("expected time %v, lstat time %v", newTime, newStat.ModTime())
 
- 	}
 
- 	if underlyingStat, err := underlying.Lstat("file"); err != nil {
 
- 		t.Fatal(err)
 
- 	} else if !underlyingStat.ModTime().Equal(oldStat.ModTime()) {
 
- 		t.Errorf("expected time %v, lstat time %v", oldStat.ModTime(), underlyingStat.ModTime())
 
- 	}
 
- 	fd, err := mtimefs.Open("file")
 
- 	if err != nil {
 
- 		t.Fatal(err)
 
- 	}
 
- 	defer fd.Close()
 
- 	info, err := fd.Stat()
 
- 	if err != nil {
 
- 		t.Fatal(err)
 
- 	}
 
- 	if !info.ModTime().Equal(newTime) {
 
- 		t.Errorf("expected time %v, lstat time %v", newTime, info.ModTime())
 
- 	}
 
- }
 
- func TestMtimeFSInsensitive(t *testing.T) {
 
- 	if build.IsDarwin || build.IsWindows {
 
- 		// blatantly assume file systems here are case insensitive. Might be
 
- 		// a spurious failure on oddly configured systems.
 
- 	} else {
 
- 		t.Skip("need case insensitive FS")
 
- 	}
 
- 	theTest := func(t *testing.T, fs *mtimeFS, shouldSucceed bool) {
 
- 		fs.RemoveAll("testdata")
 
- 		defer fs.RemoveAll("testdata")
 
- 		fs.Mkdir("testdata", 0o755)
 
- 		WriteFile(fs, "testdata/FiLe", []byte("hello"), 0o644)
 
- 		// a random time with nanosecond precision
 
- 		testTime := time.Unix(1234567890, 123456789)
 
- 		// Do one call that gets struck by an exceptionally evil Chtimes, with a
 
- 		// different case from what is on disk.
 
- 		fs.chtimes = evilChtimes
 
- 		if err := fs.Chtimes("testdata/fIlE", testTime, testTime); err != nil {
 
- 			t.Error("Should not have failed:", err)
 
- 		}
 
- 		// Check that we get back the mtime we set, if we were supposed to succeed.
 
- 		info, err := fs.Lstat("testdata/FILE")
 
- 		if err != nil {
 
- 			t.Error("Lstat shouldn't fail:", err)
 
- 		} else if info.ModTime().Equal(testTime) != shouldSucceed {
 
- 			t.Errorf("Time mismatch; got %v, comparison %v, expected equal=%v", info.ModTime(), testTime, shouldSucceed)
 
- 		}
 
- 	}
 
- 	// The test should fail with a case sensitive mtimefs
 
- 	t.Run("with case sensitive mtimefs", func(t *testing.T) {
 
- 		theTest(t, newMtimeFS(t.TempDir(), make(mapStore)), false)
 
- 	})
 
- 	// And succeed with a case insensitive one.
 
- 	t.Run("with case insensitive mtimefs", func(t *testing.T) {
 
- 		theTest(t, newMtimeFS(t.TempDir(), make(mapStore), WithCaseInsensitivity(true)), true)
 
- 	})
 
- }
 
- // The mapStore is a simple database
 
- type mapStore map[string][]byte
 
- func (s mapStore) PutBytes(key string, data []byte) error {
 
- 	s[key] = data
 
- 	return nil
 
- }
 
- func (s mapStore) Bytes(key string) (data []byte, ok bool, err error) {
 
- 	data, ok = s[key]
 
- 	return
 
- }
 
- func (s mapStore) Delete(key string) error {
 
- 	delete(s, key)
 
- 	return nil
 
- }
 
- // failChtimes does nothing, and fails
 
- func failChtimes(_ string, _, _ time.Time) error {
 
- 	return errors.New("no")
 
- }
 
- // evilChtimes will set an mtime that's 300 days in the future of what was
 
- // asked for, and truncate the time to the closest hour.
 
- func evilChtimes(name string, mtime, atime time.Time) error {
 
- 	return os.Chtimes(name, mtime.Add(300*time.Hour).Truncate(time.Hour), atime.Add(300*time.Hour).Truncate(time.Hour))
 
- }
 
- func newMtimeFS(path string, db database, options ...MtimeFSOption) *mtimeFS {
 
- 	mtimefs, _ := newMtimeFSWithWalk(path, db, options...)
 
- 	return mtimefs
 
- }
 
- func newMtimeFSWithWalk(path string, db database, options ...MtimeFSOption) (*mtimeFS, *walkFilesystem) {
 
- 	wfs := NewFilesystem(FilesystemTypeBasic, path, NewMtimeOption(db, options...)).(*walkFilesystem)
 
- 	return wfs.Filesystem.(*mtimeFS), wfs
 
- }
 
 
  |