osutil_test.go 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. // Copyright (C) 2014 The Syncthing Authors.
  2. //
  3. // This Source Code Form is subject to the terms of the Mozilla Public
  4. // License, v. 2.0. If a copy of the MPL was not distributed with this file,
  5. // You can obtain one at https://mozilla.org/MPL/2.0/.
  6. package osutil_test
  7. import (
  8. "io/ioutil"
  9. "os"
  10. "path/filepath"
  11. "runtime"
  12. "strings"
  13. "testing"
  14. "github.com/syncthing/syncthing/lib/fs"
  15. "github.com/syncthing/syncthing/lib/osutil"
  16. )
  17. func TestInWriteableDir(t *testing.T) {
  18. err := os.RemoveAll("testdata")
  19. if err != nil {
  20. t.Fatal(err)
  21. }
  22. defer os.RemoveAll("testdata")
  23. fs := fs.NewFilesystem(fs.FilesystemTypeBasic, ".")
  24. os.Mkdir("testdata", 0700)
  25. os.Mkdir("testdata/rw", 0700)
  26. os.Mkdir("testdata/ro", 0500)
  27. create := func(name string) error {
  28. fd, err := os.Create(name)
  29. if err != nil {
  30. return err
  31. }
  32. fd.Close()
  33. return nil
  34. }
  35. // These should succeed
  36. err = osutil.InWritableDir(create, fs, "testdata/file")
  37. if err != nil {
  38. t.Error("testdata/file:", err)
  39. }
  40. err = osutil.InWritableDir(create, fs, "testdata/rw/foo")
  41. if err != nil {
  42. t.Error("testdata/rw/foo:", err)
  43. }
  44. err = osutil.InWritableDir(os.Remove, fs, "testdata/rw/foo")
  45. if err != nil {
  46. t.Error("testdata/rw/foo:", err)
  47. }
  48. err = osutil.InWritableDir(create, fs, "testdata/ro/foo")
  49. if err != nil {
  50. t.Error("testdata/ro/foo:", err)
  51. }
  52. err = osutil.InWritableDir(os.Remove, fs, "testdata/ro/foo")
  53. if err != nil {
  54. t.Error("testdata/ro/foo:", err)
  55. }
  56. // These should not
  57. err = osutil.InWritableDir(create, fs, "testdata/nonexistent/foo")
  58. if err == nil {
  59. t.Error("testdata/nonexistent/foo returned nil error")
  60. }
  61. err = osutil.InWritableDir(create, fs, "testdata/file/foo")
  62. if err == nil {
  63. t.Error("testdata/file/foo returned nil error")
  64. }
  65. }
  66. func TestInWritableDirWindowsRemove(t *testing.T) {
  67. // os.Remove should remove read only things on windows
  68. if runtime.GOOS != "windows" {
  69. t.Skipf("Tests not required")
  70. return
  71. }
  72. err := os.RemoveAll("testdata")
  73. if err != nil {
  74. t.Fatal(err)
  75. }
  76. defer os.Chmod("testdata/windows/ro/readonlynew", 0700)
  77. defer os.RemoveAll("testdata")
  78. create := func(name string) error {
  79. fd, err := os.Create(name)
  80. if err != nil {
  81. return err
  82. }
  83. fd.Close()
  84. return nil
  85. }
  86. os.Mkdir("testdata", 0700)
  87. os.Mkdir("testdata/windows", 0500)
  88. os.Mkdir("testdata/windows/ro", 0500)
  89. create("testdata/windows/ro/readonly")
  90. os.Chmod("testdata/windows/ro/readonly", 0500)
  91. fs := fs.NewFilesystem(fs.FilesystemTypeBasic, ".")
  92. for _, path := range []string{"testdata/windows/ro/readonly", "testdata/windows/ro", "testdata/windows"} {
  93. err := osutil.InWritableDir(os.Remove, fs, path)
  94. if err != nil {
  95. t.Errorf("Unexpected error %s: %s", path, err)
  96. }
  97. }
  98. }
  99. func TestInWritableDirWindowsRemoveAll(t *testing.T) {
  100. // os.RemoveAll should remove read only things on windows
  101. if runtime.GOOS != "windows" {
  102. t.Skipf("Tests not required")
  103. return
  104. }
  105. err := os.RemoveAll("testdata")
  106. if err != nil {
  107. t.Fatal(err)
  108. }
  109. defer os.Chmod("testdata/windows/ro/readonlynew", 0700)
  110. defer os.RemoveAll("testdata")
  111. create := func(name string) error {
  112. fd, err := os.Create(name)
  113. if err != nil {
  114. return err
  115. }
  116. fd.Close()
  117. return nil
  118. }
  119. os.Mkdir("testdata", 0700)
  120. os.Mkdir("testdata/windows", 0500)
  121. os.Mkdir("testdata/windows/ro", 0500)
  122. create("testdata/windows/ro/readonly")
  123. os.Chmod("testdata/windows/ro/readonly", 0500)
  124. if err := os.RemoveAll("testdata/windows"); err != nil {
  125. t.Errorf("Unexpected error: %s", err)
  126. }
  127. }
  128. func TestInWritableDirWindowsRename(t *testing.T) {
  129. if runtime.GOOS != "windows" {
  130. t.Skipf("Tests not required")
  131. return
  132. }
  133. err := os.RemoveAll("testdata")
  134. if err != nil {
  135. t.Fatal(err)
  136. }
  137. defer os.Chmod("testdata/windows/ro/readonlynew", 0700)
  138. defer os.RemoveAll("testdata")
  139. create := func(name string) error {
  140. fd, err := os.Create(name)
  141. if err != nil {
  142. return err
  143. }
  144. fd.Close()
  145. return nil
  146. }
  147. os.Mkdir("testdata", 0700)
  148. os.Mkdir("testdata/windows", 0500)
  149. os.Mkdir("testdata/windows/ro", 0500)
  150. create("testdata/windows/ro/readonly")
  151. os.Chmod("testdata/windows/ro/readonly", 0500)
  152. fs := fs.NewFilesystem(fs.FilesystemTypeBasic, ".")
  153. for _, path := range []string{"testdata/windows/ro/readonly", "testdata/windows/ro", "testdata/windows"} {
  154. err := os.Rename(path, path+"new")
  155. if err == nil {
  156. t.Skipf("seem like this test doesn't work here")
  157. return
  158. }
  159. }
  160. rename := func(path string) error {
  161. return osutil.RenameOrCopy(fs, fs, path, path+"new")
  162. }
  163. for _, path := range []string{"testdata/windows/ro/readonly", "testdata/windows/ro", "testdata/windows"} {
  164. err := osutil.InWritableDir(rename, fs, path)
  165. if err != nil {
  166. t.Errorf("Unexpected error %s: %s", path, err)
  167. }
  168. _, err = os.Stat(path + "new")
  169. if err != nil {
  170. t.Errorf("Unexpected error %s: %s", path, err)
  171. }
  172. }
  173. }
  174. func TestIsDeleted(t *testing.T) {
  175. type tc struct {
  176. path string
  177. isDel bool
  178. }
  179. cases := []tc{
  180. {"del", true},
  181. {"del.file", false},
  182. {"del/del", true},
  183. {"file", false},
  184. {"linkToFile", false},
  185. {"linkToDel", false},
  186. {"linkToDir", false},
  187. {"linkToDir/file", true},
  188. {"file/behindFile", true},
  189. {"dir", false},
  190. {"dir.file", false},
  191. {"dir/file", false},
  192. {"dir/del", true},
  193. {"dir/del/del", true},
  194. {"del/del/del", true},
  195. }
  196. testFs := fs.NewFilesystem(fs.FilesystemTypeBasic, "testdata")
  197. testFs.MkdirAll("dir", 0777)
  198. for _, f := range []string{"file", "del.file", "dir.file", "dir/file"} {
  199. fd, err := testFs.Create(f)
  200. if err != nil {
  201. t.Fatal(err)
  202. }
  203. fd.Close()
  204. }
  205. if runtime.GOOS != "windows" {
  206. // Can't create unreadable dir on windows
  207. testFs.MkdirAll("inacc", 0777)
  208. if err := testFs.Chmod("inacc", 0000); err == nil {
  209. if _, err := testFs.Lstat("inacc/file"); fs.IsPermission(err) {
  210. // May fail e.g. if tests are run as root -> just skip
  211. cases = append(cases, tc{"inacc", false}, tc{"inacc/file", false})
  212. }
  213. }
  214. }
  215. for _, n := range []string{"Dir", "File", "Del"} {
  216. if err := osutil.DebugSymlinkForTestsOnly(filepath.Join(testFs.URI(), strings.ToLower(n)), filepath.Join(testFs.URI(), "linkTo"+n)); err != nil {
  217. if runtime.GOOS == "windows" {
  218. t.Skip("Symlinks aren't working")
  219. }
  220. t.Fatal(err)
  221. }
  222. }
  223. for _, c := range cases {
  224. if osutil.IsDeleted(testFs, c.path) != c.isDel {
  225. t.Errorf("IsDeleted(%v) != %v", c.path, c.isDel)
  226. }
  227. }
  228. testFs.Chmod("inacc", 0777)
  229. os.RemoveAll("testdata")
  230. }
  231. func TestRenameOrCopy(t *testing.T) {
  232. mustTempDir := func() string {
  233. t.Helper()
  234. tmpDir, err := ioutil.TempDir("", "")
  235. if err != nil {
  236. t.Fatal(err)
  237. }
  238. return tmpDir
  239. }
  240. sameFs := fs.NewFilesystem(fs.FilesystemTypeBasic, mustTempDir())
  241. tests := []struct {
  242. src fs.Filesystem
  243. dst fs.Filesystem
  244. file string
  245. }{
  246. {
  247. src: sameFs,
  248. dst: sameFs,
  249. file: "file",
  250. },
  251. {
  252. src: fs.NewFilesystem(fs.FilesystemTypeBasic, mustTempDir()),
  253. dst: fs.NewFilesystem(fs.FilesystemTypeBasic, mustTempDir()),
  254. file: "file",
  255. },
  256. {
  257. src: fs.NewFilesystem(fs.FilesystemTypeFake, `fake://fake/?files=1&seed=42`),
  258. dst: fs.NewFilesystem(fs.FilesystemTypeBasic, mustTempDir()),
  259. file: osutil.NativeFilename(`05/7a/4d52f284145b9fe8`),
  260. },
  261. }
  262. for _, test := range tests {
  263. content := test.src.URI()
  264. if _, err := test.src.Lstat(test.file); err != nil {
  265. if !fs.IsNotExist(err) {
  266. t.Fatal(err)
  267. }
  268. if fd, err := test.src.Create(test.file); err != nil {
  269. t.Fatal(err)
  270. } else {
  271. if _, err := fd.Write([]byte(test.src.URI())); err != nil {
  272. t.Fatal(err)
  273. }
  274. _ = fd.Close()
  275. }
  276. } else {
  277. fd, err := test.src.Open(test.file)
  278. if err != nil {
  279. t.Fatal(err)
  280. }
  281. buf, err := ioutil.ReadAll(fd)
  282. if err != nil {
  283. t.Fatal(err)
  284. }
  285. _ = fd.Close()
  286. content = string(buf)
  287. }
  288. err := osutil.RenameOrCopy(test.src, test.dst, test.file, "new")
  289. if err != nil {
  290. t.Fatal(err)
  291. }
  292. if fd, err := test.dst.Open("new"); err != nil {
  293. t.Fatal(err)
  294. } else {
  295. if buf, err := ioutil.ReadAll(fd); err != nil {
  296. t.Fatal(err)
  297. } else if string(buf) != content {
  298. t.Fatalf("expected %s got %s", content, string(buf))
  299. }
  300. }
  301. }
  302. }