symlink_test.go 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  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. // +build integration
  7. package integration
  8. import (
  9. "log"
  10. "os"
  11. "testing"
  12. "github.com/syncthing/syncthing/lib/config"
  13. "github.com/syncthing/syncthing/lib/protocol"
  14. "github.com/syncthing/syncthing/lib/rc"
  15. )
  16. func TestSymlinks(t *testing.T) {
  17. if !symlinksSupported() {
  18. t.Skip("symlinks unsupported")
  19. }
  20. // Use no versioning
  21. id, _ := protocol.DeviceIDFromString(id2)
  22. cfg, _ := config.Load("h2/config.xml", id)
  23. fld := cfg.Folders()["default"]
  24. fld.Versioning = config.VersioningConfiguration{}
  25. cfg.SetFolder(fld)
  26. cfg.Save()
  27. testSymlinks(t)
  28. }
  29. func TestSymlinksSimpleVersioning(t *testing.T) {
  30. if !symlinksSupported() {
  31. t.Skip("symlinks unsupported")
  32. }
  33. // Use simple versioning
  34. id, _ := protocol.DeviceIDFromString(id2)
  35. cfg, _ := config.Load("h2/config.xml", id)
  36. fld := cfg.Folders()["default"]
  37. fld.Versioning = config.VersioningConfiguration{
  38. Type: "simple",
  39. Params: map[string]string{"keep": "5"},
  40. }
  41. cfg.SetFolder(fld)
  42. cfg.Save()
  43. testSymlinks(t)
  44. }
  45. func TestSymlinksStaggeredVersioning(t *testing.T) {
  46. if !symlinksSupported() {
  47. t.Skip("symlinks unsupported")
  48. }
  49. // Use staggered versioning
  50. id, _ := protocol.DeviceIDFromString(id2)
  51. cfg, _ := config.Load("h2/config.xml", id)
  52. fld := cfg.Folders()["default"]
  53. fld.Versioning = config.VersioningConfiguration{
  54. Type: "staggered",
  55. }
  56. cfg.SetFolder(fld)
  57. cfg.Save()
  58. testSymlinks(t)
  59. }
  60. func testSymlinks(t *testing.T) {
  61. log.Println("Cleaning...")
  62. err := removeAll("s1", "s2", "h1/index*", "h2/index*")
  63. if err != nil {
  64. t.Fatal(err)
  65. }
  66. log.Println("Generating files...")
  67. err = generateFiles("s1", 100, 20, "../LICENSE")
  68. if err != nil {
  69. t.Fatal(err)
  70. }
  71. // A file that we will replace with a symlink later
  72. fd, err := os.Create("s1/fileToReplace")
  73. if err != nil {
  74. t.Fatal(err)
  75. }
  76. fd.Close()
  77. // A directory that we will replace with a symlink later
  78. err = os.Mkdir("s1/dirToReplace", 0755)
  79. if err != nil {
  80. t.Fatal(err)
  81. }
  82. // A file and a symlink to that file
  83. fd, err = os.Create("s1/file")
  84. if err != nil {
  85. t.Fatal(err)
  86. }
  87. fd.Close()
  88. err = os.Symlink("file", "s1/fileLink")
  89. if err != nil {
  90. log.Fatal(err)
  91. }
  92. // A directory and a symlink to that directory
  93. err = os.Mkdir("s1/dir", 0755)
  94. if err != nil {
  95. t.Fatal(err)
  96. }
  97. err = os.Symlink("dir", "s1/dirLink")
  98. if err != nil {
  99. log.Fatal(err)
  100. }
  101. // A link to something in the repo that does not exist
  102. err = os.Symlink("does/not/exist", "s1/noneLink")
  103. if err != nil {
  104. log.Fatal(err)
  105. }
  106. // A link we will replace with a file later
  107. err = os.Symlink("does/not/exist", "s1/repFileLink")
  108. if err != nil {
  109. log.Fatal(err)
  110. }
  111. // A link we will replace with a directory later
  112. err = os.Symlink("does/not/exist", "s1/repDirLink")
  113. if err != nil {
  114. log.Fatal(err)
  115. }
  116. // A link we will remove later
  117. err = os.Symlink("does/not/exist", "s1/removeLink")
  118. if err != nil {
  119. log.Fatal(err)
  120. }
  121. // Verify that the files and symlinks sync to the other side
  122. sender := startInstance(t, 1)
  123. defer checkedStop(t, sender)
  124. receiver := startInstance(t, 2)
  125. defer checkedStop(t, receiver)
  126. sender.ResumeAll()
  127. receiver.ResumeAll()
  128. log.Println("Syncing...")
  129. rc.AwaitSync("default", sender, receiver)
  130. log.Println("Comparing directories...")
  131. err = compareDirectories("s1", "s2")
  132. if err != nil {
  133. t.Fatal(err)
  134. }
  135. log.Println("Making some changes...")
  136. // Remove one symlink
  137. err = os.Remove("s1/fileLink")
  138. if err != nil {
  139. log.Fatal(err)
  140. }
  141. // Change the target of another
  142. err = os.Remove("s1/dirLink")
  143. if err != nil {
  144. log.Fatal(err)
  145. }
  146. err = os.Symlink("file", "s1/dirLink")
  147. if err != nil {
  148. log.Fatal(err)
  149. }
  150. // Replace one with a file
  151. err = os.Remove("s1/repFileLink")
  152. if err != nil {
  153. log.Fatal(err)
  154. }
  155. fd, err = os.Create("s1/repFileLink")
  156. if err != nil {
  157. log.Fatal(err)
  158. }
  159. fd.Close()
  160. // Replace one with a directory
  161. err = os.Remove("s1/repDirLink")
  162. if err != nil {
  163. log.Fatal(err)
  164. }
  165. err = os.Mkdir("s1/repDirLink", 0755)
  166. if err != nil {
  167. log.Fatal(err)
  168. }
  169. // Replace a file with a symlink
  170. err = os.Remove("s1/fileToReplace")
  171. if err != nil {
  172. log.Fatal(err)
  173. }
  174. err = os.Symlink("somewhere/non/existent", "s1/fileToReplace")
  175. if err != nil {
  176. log.Fatal(err)
  177. }
  178. // Replace a directory with a symlink
  179. err = os.RemoveAll("s1/dirToReplace")
  180. if err != nil {
  181. log.Fatal(err)
  182. }
  183. err = os.Symlink("somewhere/non/existent", "s1/dirToReplace")
  184. if err != nil {
  185. log.Fatal(err)
  186. }
  187. // Remove a broken symlink
  188. err = os.Remove("s1/removeLink")
  189. if err != nil {
  190. log.Fatal(err)
  191. }
  192. // Sync these changes and recheck
  193. log.Println("Syncing...")
  194. if err := sender.Rescan("default"); err != nil {
  195. t.Fatal(err)
  196. }
  197. rc.AwaitSync("default", sender, receiver)
  198. log.Println("Comparing directories...")
  199. err = compareDirectories("s1", "s2")
  200. if err != nil {
  201. t.Fatal(err)
  202. }
  203. }