2014-11-16 21:13:20 +01:00
|
|
|
// Copyright (C) 2014 The Syncthing Authors.
|
2014-09-29 21:43:32 +02:00
|
|
|
//
|
2015-03-07 21:36:35 +01:00
|
|
|
// 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,
|
2017-02-09 07:52:18 +01:00
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
2014-08-19 12:43:50 +02:00
|
|
|
|
|
|
|
package osutil_test
|
|
|
|
|
2015-03-07 14:35:29 +01:00
|
|
|
import (
|
2021-11-22 08:59:47 +01:00
|
|
|
"io"
|
2015-03-07 14:35:29 +01:00
|
|
|
"os"
|
2018-05-10 21:39:33 +02:00
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
2015-03-07 14:35:29 +01:00
|
|
|
"testing"
|
|
|
|
|
2022-07-28 19:36:39 +02:00
|
|
|
"github.com/syncthing/syncthing/lib/build"
|
2017-08-19 14:36:56 +00:00
|
|
|
"github.com/syncthing/syncthing/lib/fs"
|
2015-08-06 11:29:25 +02:00
|
|
|
"github.com/syncthing/syncthing/lib/osutil"
|
2015-03-07 14:35:29 +01:00
|
|
|
)
|
|
|
|
|
2018-05-10 21:39:33 +02:00
|
|
|
func TestIsDeleted(t *testing.T) {
|
|
|
|
type tc struct {
|
|
|
|
path string
|
|
|
|
isDel bool
|
|
|
|
}
|
|
|
|
cases := []tc{
|
|
|
|
{"del", true},
|
|
|
|
{"del.file", false},
|
2020-06-14 20:09:37 +02:00
|
|
|
{filepath.Join("del", "del"), true},
|
2018-05-10 21:39:33 +02:00
|
|
|
{"file", false},
|
|
|
|
{"linkToFile", false},
|
|
|
|
{"linkToDel", false},
|
|
|
|
{"linkToDir", false},
|
2020-06-14 20:09:37 +02:00
|
|
|
{filepath.Join("linkToDir", "file"), true},
|
|
|
|
{filepath.Join("file", "behindFile"), true},
|
2018-05-10 21:39:33 +02:00
|
|
|
{"dir", false},
|
|
|
|
{"dir.file", false},
|
2020-06-14 20:09:37 +02:00
|
|
|
{filepath.Join("dir", "file"), false},
|
|
|
|
{filepath.Join("dir", "del"), true},
|
|
|
|
{filepath.Join("dir", "del", "del"), true},
|
|
|
|
{filepath.Join("del", "del", "del"), true},
|
2018-05-10 21:39:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
testFs := fs.NewFilesystem(fs.FilesystemTypeBasic, "testdata")
|
|
|
|
|
2019-02-02 12:16:27 +01:00
|
|
|
testFs.MkdirAll("dir", 0777)
|
2020-06-14 20:09:37 +02:00
|
|
|
for _, f := range []string{"file", "del.file", "dir.file", filepath.Join("dir", "file")} {
|
2018-05-10 21:39:33 +02:00
|
|
|
fd, err := testFs.Create(f)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
fd.Close()
|
|
|
|
}
|
2022-07-28 19:36:39 +02:00
|
|
|
if !build.IsWindows {
|
2018-05-10 21:39:33 +02:00
|
|
|
// Can't create unreadable dir on windows
|
2019-02-02 12:16:27 +01:00
|
|
|
testFs.MkdirAll("inacc", 0777)
|
2018-05-10 21:39:33 +02:00
|
|
|
if err := testFs.Chmod("inacc", 0000); err == nil {
|
2020-06-14 20:09:37 +02:00
|
|
|
if _, err := testFs.Lstat(filepath.Join("inacc", "file")); fs.IsPermission(err) {
|
2018-05-10 21:39:33 +02:00
|
|
|
// May fail e.g. if tests are run as root -> just skip
|
2020-06-14 20:09:37 +02:00
|
|
|
cases = append(cases, tc{"inacc", false}, tc{filepath.Join("inacc", "file"), false})
|
2018-05-10 21:39:33 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, n := range []string{"Dir", "File", "Del"} {
|
2020-07-28 11:13:15 +02:00
|
|
|
if err := fs.DebugSymlinkForTestsOnly(testFs, testFs, strings.ToLower(n), "linkTo"+n); err != nil {
|
2022-07-28 19:36:39 +02:00
|
|
|
if build.IsWindows {
|
2018-05-10 21:39:33 +02:00
|
|
|
t.Skip("Symlinks aren't working")
|
|
|
|
}
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range cases {
|
|
|
|
if osutil.IsDeleted(testFs, c.path) != c.isDel {
|
|
|
|
t.Errorf("IsDeleted(%v) != %v", c.path, c.isDel)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-02 12:16:27 +01:00
|
|
|
testFs.Chmod("inacc", 0777)
|
2018-05-10 21:39:33 +02:00
|
|
|
os.RemoveAll("testdata")
|
|
|
|
}
|
2019-04-28 23:30:16 +01:00
|
|
|
|
|
|
|
func TestRenameOrCopy(t *testing.T) {
|
2022-04-15 11:44:06 +08:00
|
|
|
sameFs := fs.NewFilesystem(fs.FilesystemTypeBasic, t.TempDir())
|
2019-04-28 23:30:16 +01:00
|
|
|
tests := []struct {
|
|
|
|
src fs.Filesystem
|
|
|
|
dst fs.Filesystem
|
|
|
|
file string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
src: sameFs,
|
|
|
|
dst: sameFs,
|
|
|
|
file: "file",
|
|
|
|
},
|
|
|
|
{
|
2022-04-15 11:44:06 +08:00
|
|
|
src: fs.NewFilesystem(fs.FilesystemTypeBasic, t.TempDir()),
|
|
|
|
dst: fs.NewFilesystem(fs.FilesystemTypeBasic, t.TempDir()),
|
2019-04-28 23:30:16 +01:00
|
|
|
file: "file",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
src: fs.NewFilesystem(fs.FilesystemTypeFake, `fake://fake/?files=1&seed=42`),
|
2022-04-15 11:44:06 +08:00
|
|
|
dst: fs.NewFilesystem(fs.FilesystemTypeBasic, t.TempDir()),
|
2019-04-28 23:30:16 +01:00
|
|
|
file: osutil.NativeFilename(`05/7a/4d52f284145b9fe8`),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
content := test.src.URI()
|
|
|
|
if _, err := test.src.Lstat(test.file); err != nil {
|
|
|
|
if !fs.IsNotExist(err) {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if fd, err := test.src.Create(test.file); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
} else {
|
|
|
|
if _, err := fd.Write([]byte(test.src.URI())); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
_ = fd.Close()
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fd, err := test.src.Open(test.file)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2021-11-22 08:59:47 +01:00
|
|
|
buf, err := io.ReadAll(fd)
|
2019-04-28 23:30:16 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
_ = fd.Close()
|
|
|
|
content = string(buf)
|
|
|
|
}
|
|
|
|
|
2020-06-18 07:15:47 +01:00
|
|
|
err := osutil.RenameOrCopy(fs.CopyRangeMethodStandard, test.src, test.dst, test.file, "new")
|
2019-04-28 23:30:16 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if fd, err := test.dst.Open("new"); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
} else {
|
2022-04-15 11:44:06 +08:00
|
|
|
t.Cleanup(func() {
|
|
|
|
_ = fd.Close()
|
|
|
|
})
|
|
|
|
|
2021-11-22 08:59:47 +01:00
|
|
|
if buf, err := io.ReadAll(fd); err != nil {
|
2019-04-28 23:30:16 +01:00
|
|
|
t.Fatal(err)
|
|
|
|
} else if string(buf) != content {
|
|
|
|
t.Fatalf("expected %s got %s", content, string(buf))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|