2014-11-16 20:13:20 +00:00
|
|
|
// Copyright (C) 2014 The Syncthing Authors.
|
2014-09-29 19:43:32 +00:00
|
|
|
//
|
2015-03-07 20:36:35 +00: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 06:52:18 +00:00
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
2014-08-19 10:43:50 +00:00
|
|
|
|
|
|
|
package osutil_test
|
|
|
|
|
2015-03-07 13:35:29 +00:00
|
|
|
import (
|
2019-04-28 22:30:16 +00:00
|
|
|
"io/ioutil"
|
2015-03-07 13:35:29 +00:00
|
|
|
"os"
|
2018-05-10 19:39:33 +00:00
|
|
|
"path/filepath"
|
2015-04-16 21:07:04 +00:00
|
|
|
"runtime"
|
2018-05-10 19:39:33 +00:00
|
|
|
"strings"
|
2015-03-07 13:35:29 +00:00
|
|
|
"testing"
|
|
|
|
|
2017-08-19 14:36:56 +00:00
|
|
|
"github.com/syncthing/syncthing/lib/fs"
|
2015-08-06 09:29:25 +00:00
|
|
|
"github.com/syncthing/syncthing/lib/osutil"
|
2015-03-07 13:35:29 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestInWriteableDir(t *testing.T) {
|
|
|
|
err := os.RemoveAll("testdata")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll("testdata")
|
|
|
|
|
2017-08-19 14:36:56 +00:00
|
|
|
fs := fs.NewFilesystem(fs.FilesystemTypeBasic, ".")
|
|
|
|
|
2019-02-02 11:16:27 +00:00
|
|
|
os.Mkdir("testdata", 0700)
|
|
|
|
os.Mkdir("testdata/rw", 0700)
|
|
|
|
os.Mkdir("testdata/ro", 0500)
|
2015-03-07 13:35:29 +00:00
|
|
|
|
|
|
|
create := func(name string) error {
|
|
|
|
fd, err := os.Create(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fd.Close()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// These should succeed
|
|
|
|
|
2017-08-19 14:36:56 +00:00
|
|
|
err = osutil.InWritableDir(create, fs, "testdata/file")
|
2015-03-07 13:35:29 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Error("testdata/file:", err)
|
|
|
|
}
|
2017-08-19 14:36:56 +00:00
|
|
|
err = osutil.InWritableDir(create, fs, "testdata/rw/foo")
|
2015-03-07 13:35:29 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Error("testdata/rw/foo:", err)
|
|
|
|
}
|
2017-08-19 14:36:56 +00:00
|
|
|
err = osutil.InWritableDir(os.Remove, fs, "testdata/rw/foo")
|
2015-03-07 13:35:29 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Error("testdata/rw/foo:", err)
|
|
|
|
}
|
|
|
|
|
2017-08-19 14:36:56 +00:00
|
|
|
err = osutil.InWritableDir(create, fs, "testdata/ro/foo")
|
2015-03-07 13:35:29 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Error("testdata/ro/foo:", err)
|
|
|
|
}
|
2017-08-19 14:36:56 +00:00
|
|
|
err = osutil.InWritableDir(os.Remove, fs, "testdata/ro/foo")
|
2015-03-07 13:35:29 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Error("testdata/ro/foo:", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// These should not
|
|
|
|
|
2017-08-19 14:36:56 +00:00
|
|
|
err = osutil.InWritableDir(create, fs, "testdata/nonexistent/foo")
|
2015-03-07 13:35:29 +00:00
|
|
|
if err == nil {
|
|
|
|
t.Error("testdata/nonexistent/foo returned nil error")
|
|
|
|
}
|
2017-08-19 14:36:56 +00:00
|
|
|
err = osutil.InWritableDir(create, fs, "testdata/file/foo")
|
2015-03-07 13:35:29 +00:00
|
|
|
if err == nil {
|
|
|
|
t.Error("testdata/file/foo returned nil error")
|
|
|
|
}
|
|
|
|
}
|
2015-04-16 21:07:04 +00:00
|
|
|
|
|
|
|
func TestInWritableDirWindowsRemove(t *testing.T) {
|
2016-08-16 10:01:58 +00:00
|
|
|
// os.Remove should remove read only things on windows
|
|
|
|
|
2015-04-16 21:07:04 +00:00
|
|
|
if runtime.GOOS != "windows" {
|
|
|
|
t.Skipf("Tests not required")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err := os.RemoveAll("testdata")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-02-02 11:16:27 +00:00
|
|
|
defer os.Chmod("testdata/windows/ro/readonlynew", 0700)
|
2015-04-16 21:07:04 +00:00
|
|
|
defer os.RemoveAll("testdata")
|
|
|
|
|
|
|
|
create := func(name string) error {
|
|
|
|
fd, err := os.Create(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fd.Close()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-02-02 11:16:27 +00:00
|
|
|
os.Mkdir("testdata", 0700)
|
2015-04-16 21:07:04 +00:00
|
|
|
|
2019-02-02 11:16:27 +00:00
|
|
|
os.Mkdir("testdata/windows", 0500)
|
|
|
|
os.Mkdir("testdata/windows/ro", 0500)
|
|
|
|
create("testdata/windows/ro/readonly")
|
|
|
|
os.Chmod("testdata/windows/ro/readonly", 0500)
|
2015-04-16 21:07:04 +00:00
|
|
|
|
2017-08-19 14:36:56 +00:00
|
|
|
fs := fs.NewFilesystem(fs.FilesystemTypeBasic, ".")
|
|
|
|
|
2015-04-16 21:07:04 +00:00
|
|
|
for _, path := range []string{"testdata/windows/ro/readonly", "testdata/windows/ro", "testdata/windows"} {
|
2017-08-19 14:36:56 +00:00
|
|
|
err := osutil.InWritableDir(os.Remove, fs, path)
|
2016-08-16 10:01:58 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error %s: %s", path, err)
|
2015-04-16 21:07:04 +00:00
|
|
|
}
|
|
|
|
}
|
2016-08-16 10:01:58 +00:00
|
|
|
}
|
2015-04-16 21:07:04 +00:00
|
|
|
|
2016-08-16 10:01:58 +00:00
|
|
|
func TestInWritableDirWindowsRemoveAll(t *testing.T) {
|
|
|
|
// os.RemoveAll should remove read only things on windows
|
|
|
|
|
|
|
|
if runtime.GOOS != "windows" {
|
|
|
|
t.Skipf("Tests not required")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err := os.RemoveAll("testdata")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-02-02 11:16:27 +00:00
|
|
|
defer os.Chmod("testdata/windows/ro/readonlynew", 0700)
|
|
|
|
defer os.RemoveAll("testdata")
|
2016-08-16 10:01:58 +00:00
|
|
|
|
|
|
|
create := func(name string) error {
|
|
|
|
fd, err := os.Create(name)
|
2015-04-16 21:07:04 +00:00
|
|
|
if err != nil {
|
2016-08-16 10:01:58 +00:00
|
|
|
return err
|
2015-04-16 21:07:04 +00:00
|
|
|
}
|
2016-08-16 10:01:58 +00:00
|
|
|
fd.Close()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-02-02 11:16:27 +00:00
|
|
|
os.Mkdir("testdata", 0700)
|
2016-08-16 10:01:58 +00:00
|
|
|
|
2019-02-02 11:16:27 +00:00
|
|
|
os.Mkdir("testdata/windows", 0500)
|
|
|
|
os.Mkdir("testdata/windows/ro", 0500)
|
|
|
|
create("testdata/windows/ro/readonly")
|
|
|
|
os.Chmod("testdata/windows/ro/readonly", 0500)
|
2016-08-16 10:01:58 +00:00
|
|
|
|
|
|
|
if err := os.RemoveAll("testdata/windows"); err != nil {
|
|
|
|
t.Errorf("Unexpected error: %s", err)
|
2015-04-16 21:07:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestInWritableDirWindowsRename(t *testing.T) {
|
|
|
|
if runtime.GOOS != "windows" {
|
|
|
|
t.Skipf("Tests not required")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err := os.RemoveAll("testdata")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-02-02 11:16:27 +00:00
|
|
|
defer os.Chmod("testdata/windows/ro/readonlynew", 0700)
|
|
|
|
defer os.RemoveAll("testdata")
|
2015-04-16 21:07:04 +00:00
|
|
|
|
|
|
|
create := func(name string) error {
|
|
|
|
fd, err := os.Create(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fd.Close()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-02-02 11:16:27 +00:00
|
|
|
os.Mkdir("testdata", 0700)
|
2015-04-16 21:07:04 +00:00
|
|
|
|
2019-02-02 11:16:27 +00:00
|
|
|
os.Mkdir("testdata/windows", 0500)
|
|
|
|
os.Mkdir("testdata/windows/ro", 0500)
|
|
|
|
create("testdata/windows/ro/readonly")
|
|
|
|
os.Chmod("testdata/windows/ro/readonly", 0500)
|
2015-04-16 21:07:04 +00:00
|
|
|
|
2017-08-19 14:36:56 +00:00
|
|
|
fs := fs.NewFilesystem(fs.FilesystemTypeBasic, ".")
|
|
|
|
|
2015-04-16 21:07:04 +00:00
|
|
|
for _, path := range []string{"testdata/windows/ro/readonly", "testdata/windows/ro", "testdata/windows"} {
|
|
|
|
err := os.Rename(path, path+"new")
|
|
|
|
if err == nil {
|
2015-05-07 20:16:14 +00:00
|
|
|
t.Skipf("seem like this test doesn't work here")
|
|
|
|
return
|
2015-04-16 21:07:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rename := func(path string) error {
|
2019-04-28 22:30:16 +00:00
|
|
|
return osutil.RenameOrCopy(fs, fs, path, path+"new")
|
2015-04-16 21:07:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, path := range []string{"testdata/windows/ro/readonly", "testdata/windows/ro", "testdata/windows"} {
|
2017-08-19 14:36:56 +00:00
|
|
|
err := osutil.InWritableDir(rename, fs, path)
|
2015-04-16 21:07:04 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error %s: %s", path, err)
|
|
|
|
}
|
|
|
|
_, err = os.Stat(path + "new")
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error %s: %s", path, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-05-10 19:39:33 +00:00
|
|
|
|
|
|
|
func TestIsDeleted(t *testing.T) {
|
|
|
|
type tc struct {
|
|
|
|
path string
|
|
|
|
isDel bool
|
|
|
|
}
|
|
|
|
cases := []tc{
|
|
|
|
{"del", true},
|
|
|
|
{"del.file", false},
|
|
|
|
{"del/del", true},
|
|
|
|
{"file", false},
|
|
|
|
{"linkToFile", false},
|
|
|
|
{"linkToDel", false},
|
|
|
|
{"linkToDir", false},
|
|
|
|
{"linkToDir/file", true},
|
|
|
|
{"file/behindFile", true},
|
|
|
|
{"dir", false},
|
|
|
|
{"dir.file", false},
|
|
|
|
{"dir/file", false},
|
|
|
|
{"dir/del", true},
|
|
|
|
{"dir/del/del", true},
|
|
|
|
{"del/del/del", true},
|
|
|
|
}
|
|
|
|
|
|
|
|
testFs := fs.NewFilesystem(fs.FilesystemTypeBasic, "testdata")
|
|
|
|
|
2019-02-02 11:16:27 +00:00
|
|
|
testFs.MkdirAll("dir", 0777)
|
2018-05-10 19:39:33 +00:00
|
|
|
for _, f := range []string{"file", "del.file", "dir.file", "dir/file"} {
|
|
|
|
fd, err := testFs.Create(f)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
fd.Close()
|
|
|
|
}
|
|
|
|
if runtime.GOOS != "windows" {
|
|
|
|
// Can't create unreadable dir on windows
|
2019-02-02 11:16:27 +00:00
|
|
|
testFs.MkdirAll("inacc", 0777)
|
2018-05-10 19:39:33 +00:00
|
|
|
if err := testFs.Chmod("inacc", 0000); err == nil {
|
|
|
|
if _, err := testFs.Lstat("inacc/file"); fs.IsPermission(err) {
|
|
|
|
// May fail e.g. if tests are run as root -> just skip
|
|
|
|
cases = append(cases, tc{"inacc", false}, tc{"inacc/file", false})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, n := range []string{"Dir", "File", "Del"} {
|
|
|
|
if err := osutil.DebugSymlinkForTestsOnly(filepath.Join(testFs.URI(), strings.ToLower(n)), filepath.Join(testFs.URI(), "linkTo"+n)); err != nil {
|
|
|
|
if runtime.GOOS == "windows" {
|
|
|
|
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 11:16:27 +00:00
|
|
|
testFs.Chmod("inacc", 0777)
|
2018-05-10 19:39:33 +00:00
|
|
|
os.RemoveAll("testdata")
|
|
|
|
}
|
2019-04-28 22:30:16 +00:00
|
|
|
|
|
|
|
func TestRenameOrCopy(t *testing.T) {
|
|
|
|
mustTempDir := func() string {
|
|
|
|
t.Helper()
|
|
|
|
tmpDir, err := ioutil.TempDir("", "")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
return tmpDir
|
|
|
|
}
|
|
|
|
sameFs := fs.NewFilesystem(fs.FilesystemTypeBasic, mustTempDir())
|
|
|
|
tests := []struct {
|
|
|
|
src fs.Filesystem
|
|
|
|
dst fs.Filesystem
|
|
|
|
file string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
src: sameFs,
|
|
|
|
dst: sameFs,
|
|
|
|
file: "file",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
src: fs.NewFilesystem(fs.FilesystemTypeBasic, mustTempDir()),
|
|
|
|
dst: fs.NewFilesystem(fs.FilesystemTypeBasic, mustTempDir()),
|
|
|
|
file: "file",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
src: fs.NewFilesystem(fs.FilesystemTypeFake, `fake://fake/?files=1&seed=42`),
|
|
|
|
dst: fs.NewFilesystem(fs.FilesystemTypeBasic, mustTempDir()),
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
buf, err := ioutil.ReadAll(fd)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
_ = fd.Close()
|
|
|
|
content = string(buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
err := osutil.RenameOrCopy(test.src, test.dst, test.file, "new")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if fd, err := test.dst.Open("new"); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
} else {
|
|
|
|
if buf, err := ioutil.ReadAll(fd); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
} else if string(buf) != content {
|
|
|
|
t.Fatalf("expected %s got %s", content, string(buf))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|