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 (
|
|
|
|
"os"
|
2018-05-10 21:39:33 +02:00
|
|
|
"path/filepath"
|
2015-04-16 22:07:04 +01:00
|
|
|
"runtime"
|
2018-05-10 21:39:33 +02:00
|
|
|
"strings"
|
2015-03-07 14:35:29 +01:00
|
|
|
"testing"
|
|
|
|
|
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
|
|
|
)
|
|
|
|
|
|
|
|
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, ".")
|
|
|
|
|
2015-03-07 14:35:29 +01:00
|
|
|
os.Mkdir("testdata", 0700)
|
|
|
|
os.Mkdir("testdata/rw", 0700)
|
|
|
|
os.Mkdir("testdata/ro", 0500)
|
|
|
|
|
|
|
|
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 14:35:29 +01: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 14:35:29 +01: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 14:35:29 +01: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 14:35:29 +01: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 14:35:29 +01: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 14:35:29 +01: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 14:35:29 +01:00
|
|
|
if err == nil {
|
|
|
|
t.Error("testdata/file/foo returned nil error")
|
|
|
|
}
|
|
|
|
}
|
2015-04-16 22:07:04 +01: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 22:07:04 +01:00
|
|
|
if runtime.GOOS != "windows" {
|
|
|
|
t.Skipf("Tests not required")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err := os.RemoveAll("testdata")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2016-03-06 20:32:10 +00:00
|
|
|
defer os.Chmod("testdata/windows/ro/readonlynew", 0700)
|
2015-04-16 22:07:04 +01:00
|
|
|
defer os.RemoveAll("testdata")
|
|
|
|
|
|
|
|
create := func(name string) error {
|
|
|
|
fd, err := os.Create(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fd.Close()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
os.Mkdir("testdata", 0700)
|
|
|
|
|
|
|
|
os.Mkdir("testdata/windows", 0500)
|
|
|
|
os.Mkdir("testdata/windows/ro", 0500)
|
|
|
|
create("testdata/windows/ro/readonly")
|
|
|
|
os.Chmod("testdata/windows/ro/readonly", 0500)
|
|
|
|
|
2017-08-19 14:36:56 +00:00
|
|
|
fs := fs.NewFilesystem(fs.FilesystemTypeBasic, ".")
|
|
|
|
|
2015-04-16 22:07:04 +01: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 22:07:04 +01:00
|
|
|
}
|
|
|
|
}
|
2016-08-16 10:01:58 +00:00
|
|
|
}
|
2015-04-16 22:07:04 +01: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)
|
|
|
|
}
|
|
|
|
defer os.Chmod("testdata/windows/ro/readonlynew", 0700)
|
|
|
|
defer os.RemoveAll("testdata")
|
|
|
|
|
|
|
|
create := func(name string) error {
|
|
|
|
fd, err := os.Create(name)
|
2015-04-16 22:07:04 +01:00
|
|
|
if err != nil {
|
2016-08-16 10:01:58 +00:00
|
|
|
return err
|
2015-04-16 22:07:04 +01:00
|
|
|
}
|
2016-08-16 10:01:58 +00:00
|
|
|
fd.Close()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
os.Mkdir("testdata", 0700)
|
|
|
|
|
|
|
|
os.Mkdir("testdata/windows", 0500)
|
|
|
|
os.Mkdir("testdata/windows/ro", 0500)
|
|
|
|
create("testdata/windows/ro/readonly")
|
|
|
|
os.Chmod("testdata/windows/ro/readonly", 0500)
|
|
|
|
|
|
|
|
if err := os.RemoveAll("testdata/windows"); err != nil {
|
|
|
|
t.Errorf("Unexpected error: %s", err)
|
2015-04-16 22:07:04 +01: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)
|
|
|
|
}
|
2016-03-06 20:32:10 +00:00
|
|
|
defer os.Chmod("testdata/windows/ro/readonlynew", 0700)
|
2015-04-16 22:07:04 +01:00
|
|
|
defer os.RemoveAll("testdata")
|
|
|
|
|
|
|
|
create := func(name string) error {
|
|
|
|
fd, err := os.Create(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fd.Close()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
os.Mkdir("testdata", 0700)
|
|
|
|
|
|
|
|
os.Mkdir("testdata/windows", 0500)
|
|
|
|
os.Mkdir("testdata/windows/ro", 0500)
|
|
|
|
create("testdata/windows/ro/readonly")
|
|
|
|
os.Chmod("testdata/windows/ro/readonly", 0500)
|
|
|
|
|
2017-08-19 14:36:56 +00:00
|
|
|
fs := fs.NewFilesystem(fs.FilesystemTypeBasic, ".")
|
|
|
|
|
2015-04-16 22:07:04 +01: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 22:16:14 +02:00
|
|
|
t.Skipf("seem like this test doesn't work here")
|
|
|
|
return
|
2015-04-16 22:07:04 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rename := func(path string) error {
|
2017-08-19 14:36:56 +00:00
|
|
|
return osutil.Rename(fs, path, path+"new")
|
2015-04-16 22:07:04 +01: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 22:07:04 +01: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 21:39:33 +02: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")
|
|
|
|
|
|
|
|
testFs.MkdirAll("dir", 0777)
|
|
|
|
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
|
|
|
|
testFs.MkdirAll("inacc", 0777)
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
testFs.Chmod("inacc", 0777)
|
|
|
|
os.RemoveAll("testdata")
|
|
|
|
}
|