sharedsubtree_linux_test.go 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. // +build linux
  2. package mount // import "github.com/docker/docker/pkg/mount"
  3. import (
  4. "os"
  5. "path"
  6. "testing"
  7. "golang.org/x/sys/unix"
  8. )
  9. // nothing is propagated in or out
  10. func TestSubtreePrivate(t *testing.T) {
  11. if os.Getuid() != 0 {
  12. t.Skip("root required")
  13. }
  14. tmp := path.Join(os.TempDir(), "mount-tests")
  15. if err := os.MkdirAll(tmp, 0777); err != nil {
  16. t.Fatal(err)
  17. }
  18. defer os.RemoveAll(tmp)
  19. var (
  20. sourceDir = path.Join(tmp, "source")
  21. targetDir = path.Join(tmp, "target")
  22. outside1Dir = path.Join(tmp, "outside1")
  23. outside2Dir = path.Join(tmp, "outside2")
  24. outside1Path = path.Join(outside1Dir, "file.txt")
  25. outside2Path = path.Join(outside2Dir, "file.txt")
  26. outside1CheckPath = path.Join(targetDir, "a", "file.txt")
  27. outside2CheckPath = path.Join(sourceDir, "b", "file.txt")
  28. )
  29. if err := os.MkdirAll(path.Join(sourceDir, "a"), 0777); err != nil {
  30. t.Fatal(err)
  31. }
  32. if err := os.MkdirAll(path.Join(sourceDir, "b"), 0777); err != nil {
  33. t.Fatal(err)
  34. }
  35. if err := os.Mkdir(targetDir, 0777); err != nil {
  36. t.Fatal(err)
  37. }
  38. if err := os.Mkdir(outside1Dir, 0777); err != nil {
  39. t.Fatal(err)
  40. }
  41. if err := os.Mkdir(outside2Dir, 0777); err != nil {
  42. t.Fatal(err)
  43. }
  44. if err := createFile(outside1Path); err != nil {
  45. t.Fatal(err)
  46. }
  47. if err := createFile(outside2Path); err != nil {
  48. t.Fatal(err)
  49. }
  50. // mount the shared directory to a target
  51. if err := Mount(sourceDir, targetDir, "none", "bind,rw"); err != nil {
  52. t.Fatal(err)
  53. }
  54. defer func() {
  55. if err := Unmount(targetDir); err != nil {
  56. t.Fatal(err)
  57. }
  58. }()
  59. // next, make the target private
  60. if err := MakePrivate(targetDir); err != nil {
  61. t.Fatal(err)
  62. }
  63. defer func() {
  64. if err := Unmount(targetDir); err != nil {
  65. t.Fatal(err)
  66. }
  67. }()
  68. // mount in an outside path to a mounted path inside the _source_
  69. if err := Mount(outside1Dir, path.Join(sourceDir, "a"), "none", "bind,rw"); err != nil {
  70. t.Fatal(err)
  71. }
  72. defer func() {
  73. if err := Unmount(path.Join(sourceDir, "a")); err != nil {
  74. t.Fatal(err)
  75. }
  76. }()
  77. // check that this file _does_not_ show in the _target_
  78. if _, err := os.Stat(outside1CheckPath); err != nil && !os.IsNotExist(err) {
  79. t.Fatal(err)
  80. } else if err == nil {
  81. t.Fatalf("%q should not be visible, but is", outside1CheckPath)
  82. }
  83. // next mount outside2Dir into the _target_
  84. if err := Mount(outside2Dir, path.Join(targetDir, "b"), "none", "bind,rw"); err != nil {
  85. t.Fatal(err)
  86. }
  87. defer func() {
  88. if err := Unmount(path.Join(targetDir, "b")); err != nil {
  89. t.Fatal(err)
  90. }
  91. }()
  92. // check that this file _does_not_ show in the _source_
  93. if _, err := os.Stat(outside2CheckPath); err != nil && !os.IsNotExist(err) {
  94. t.Fatal(err)
  95. } else if err == nil {
  96. t.Fatalf("%q should not be visible, but is", outside2CheckPath)
  97. }
  98. }
  99. // Testing that when a target is a shared mount,
  100. // then child mounts propagate to the source
  101. func TestSubtreeShared(t *testing.T) {
  102. if os.Getuid() != 0 {
  103. t.Skip("root required")
  104. }
  105. tmp := path.Join(os.TempDir(), "mount-tests")
  106. if err := os.MkdirAll(tmp, 0777); err != nil {
  107. t.Fatal(err)
  108. }
  109. defer os.RemoveAll(tmp)
  110. var (
  111. sourceDir = path.Join(tmp, "source")
  112. targetDir = path.Join(tmp, "target")
  113. outsideDir = path.Join(tmp, "outside")
  114. outsidePath = path.Join(outsideDir, "file.txt")
  115. sourceCheckPath = path.Join(sourceDir, "a", "file.txt")
  116. )
  117. if err := os.MkdirAll(path.Join(sourceDir, "a"), 0777); err != nil {
  118. t.Fatal(err)
  119. }
  120. if err := os.Mkdir(targetDir, 0777); err != nil {
  121. t.Fatal(err)
  122. }
  123. if err := os.Mkdir(outsideDir, 0777); err != nil {
  124. t.Fatal(err)
  125. }
  126. if err := createFile(outsidePath); err != nil {
  127. t.Fatal(err)
  128. }
  129. // mount the source as shared
  130. if err := MakeShared(sourceDir); err != nil {
  131. t.Fatal(err)
  132. }
  133. defer func() {
  134. if err := Unmount(sourceDir); err != nil {
  135. t.Fatal(err)
  136. }
  137. }()
  138. // mount the shared directory to a target
  139. if err := Mount(sourceDir, targetDir, "none", "bind,rw"); err != nil {
  140. t.Fatal(err)
  141. }
  142. defer func() {
  143. if err := Unmount(targetDir); err != nil {
  144. t.Fatal(err)
  145. }
  146. }()
  147. // mount in an outside path to a mounted path inside the target
  148. if err := Mount(outsideDir, path.Join(targetDir, "a"), "none", "bind,rw"); err != nil {
  149. t.Fatal(err)
  150. }
  151. defer func() {
  152. if err := Unmount(path.Join(targetDir, "a")); err != nil {
  153. t.Fatal(err)
  154. }
  155. }()
  156. // NOW, check that the file from the outside directory is available in the source directory
  157. if _, err := os.Stat(sourceCheckPath); err != nil {
  158. t.Fatal(err)
  159. }
  160. }
  161. // testing that mounts to a shared source show up in the slave target,
  162. // and that mounts into a slave target do _not_ show up in the shared source
  163. func TestSubtreeSharedSlave(t *testing.T) {
  164. if os.Getuid() != 0 {
  165. t.Skip("root required")
  166. }
  167. tmp := path.Join(os.TempDir(), "mount-tests")
  168. if err := os.MkdirAll(tmp, 0777); err != nil {
  169. t.Fatal(err)
  170. }
  171. defer os.RemoveAll(tmp)
  172. var (
  173. sourceDir = path.Join(tmp, "source")
  174. targetDir = path.Join(tmp, "target")
  175. outside1Dir = path.Join(tmp, "outside1")
  176. outside2Dir = path.Join(tmp, "outside2")
  177. outside1Path = path.Join(outside1Dir, "file.txt")
  178. outside2Path = path.Join(outside2Dir, "file.txt")
  179. outside1CheckPath = path.Join(targetDir, "a", "file.txt")
  180. outside2CheckPath = path.Join(sourceDir, "b", "file.txt")
  181. )
  182. if err := os.MkdirAll(path.Join(sourceDir, "a"), 0777); err != nil {
  183. t.Fatal(err)
  184. }
  185. if err := os.MkdirAll(path.Join(sourceDir, "b"), 0777); err != nil {
  186. t.Fatal(err)
  187. }
  188. if err := os.Mkdir(targetDir, 0777); err != nil {
  189. t.Fatal(err)
  190. }
  191. if err := os.Mkdir(outside1Dir, 0777); err != nil {
  192. t.Fatal(err)
  193. }
  194. if err := os.Mkdir(outside2Dir, 0777); err != nil {
  195. t.Fatal(err)
  196. }
  197. if err := createFile(outside1Path); err != nil {
  198. t.Fatal(err)
  199. }
  200. if err := createFile(outside2Path); err != nil {
  201. t.Fatal(err)
  202. }
  203. // mount the source as shared
  204. if err := MakeShared(sourceDir); err != nil {
  205. t.Fatal(err)
  206. }
  207. defer func() {
  208. if err := Unmount(sourceDir); err != nil {
  209. t.Fatal(err)
  210. }
  211. }()
  212. // mount the shared directory to a target
  213. if err := Mount(sourceDir, targetDir, "none", "bind,rw"); err != nil {
  214. t.Fatal(err)
  215. }
  216. defer func() {
  217. if err := Unmount(targetDir); err != nil {
  218. t.Fatal(err)
  219. }
  220. }()
  221. // next, make the target slave
  222. if err := MakeSlave(targetDir); err != nil {
  223. t.Fatal(err)
  224. }
  225. defer func() {
  226. if err := Unmount(targetDir); err != nil {
  227. t.Fatal(err)
  228. }
  229. }()
  230. // mount in an outside path to a mounted path inside the _source_
  231. if err := Mount(outside1Dir, path.Join(sourceDir, "a"), "none", "bind,rw"); err != nil {
  232. t.Fatal(err)
  233. }
  234. defer func() {
  235. if err := Unmount(path.Join(sourceDir, "a")); err != nil {
  236. t.Fatal(err)
  237. }
  238. }()
  239. // check that this file _does_ show in the _target_
  240. if _, err := os.Stat(outside1CheckPath); err != nil {
  241. t.Fatal(err)
  242. }
  243. // next mount outside2Dir into the _target_
  244. if err := Mount(outside2Dir, path.Join(targetDir, "b"), "none", "bind,rw"); err != nil {
  245. t.Fatal(err)
  246. }
  247. defer func() {
  248. if err := Unmount(path.Join(targetDir, "b")); err != nil {
  249. t.Fatal(err)
  250. }
  251. }()
  252. // check that this file _does_not_ show in the _source_
  253. if _, err := os.Stat(outside2CheckPath); err != nil && !os.IsNotExist(err) {
  254. t.Fatal(err)
  255. } else if err == nil {
  256. t.Fatalf("%q should not be visible, but is", outside2CheckPath)
  257. }
  258. }
  259. func TestSubtreeUnbindable(t *testing.T) {
  260. if os.Getuid() != 0 {
  261. t.Skip("root required")
  262. }
  263. tmp := path.Join(os.TempDir(), "mount-tests")
  264. if err := os.MkdirAll(tmp, 0777); err != nil {
  265. t.Fatal(err)
  266. }
  267. defer os.RemoveAll(tmp)
  268. var (
  269. sourceDir = path.Join(tmp, "source")
  270. targetDir = path.Join(tmp, "target")
  271. )
  272. if err := os.MkdirAll(sourceDir, 0777); err != nil {
  273. t.Fatal(err)
  274. }
  275. if err := os.MkdirAll(targetDir, 0777); err != nil {
  276. t.Fatal(err)
  277. }
  278. // next, make the source unbindable
  279. if err := MakeUnbindable(sourceDir); err != nil {
  280. t.Fatal(err)
  281. }
  282. defer func() {
  283. if err := Unmount(sourceDir); err != nil {
  284. t.Fatal(err)
  285. }
  286. }()
  287. // then attempt to mount it to target. It should fail
  288. if err := Mount(sourceDir, targetDir, "none", "bind,rw"); err != nil && err != unix.EINVAL {
  289. t.Fatal(err)
  290. } else if err == nil {
  291. t.Fatalf("%q should not have been bindable", sourceDir)
  292. }
  293. defer func() {
  294. if err := Unmount(targetDir); err != nil {
  295. t.Fatal(err)
  296. }
  297. }()
  298. }
  299. func createFile(path string) error {
  300. f, err := os.Create(path)
  301. if err != nil {
  302. return err
  303. }
  304. f.WriteString("hello world!")
  305. return f.Close()
  306. }