local_test.go 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. package local
  2. import (
  3. "io/ioutil"
  4. "os"
  5. "reflect"
  6. "runtime"
  7. "strings"
  8. "testing"
  9. "github.com/docker/docker/pkg/mount"
  10. )
  11. func TestRemove(t *testing.T) {
  12. // TODO Windows: Investigate why this test fails on Windows under CI
  13. // but passes locally.
  14. if runtime.GOOS == "windows" {
  15. t.Skip("Test failing on Windows CI")
  16. }
  17. rootDir, err := ioutil.TempDir("", "local-volume-test")
  18. if err != nil {
  19. t.Fatal(err)
  20. }
  21. defer os.RemoveAll(rootDir)
  22. r, err := New(rootDir, 0, 0)
  23. if err != nil {
  24. t.Fatal(err)
  25. }
  26. vol, err := r.Create("testing", nil)
  27. if err != nil {
  28. t.Fatal(err)
  29. }
  30. if err := r.Remove(vol); err != nil {
  31. t.Fatal(err)
  32. }
  33. vol, err = r.Create("testing2", nil)
  34. if err != nil {
  35. t.Fatal(err)
  36. }
  37. if err := os.RemoveAll(vol.Path()); err != nil {
  38. t.Fatal(err)
  39. }
  40. if err := r.Remove(vol); err != nil {
  41. t.Fatal(err)
  42. }
  43. if _, err := os.Stat(vol.Path()); err != nil && !os.IsNotExist(err) {
  44. t.Fatal("volume dir not removed")
  45. }
  46. if l, _ := r.List(); len(l) != 0 {
  47. t.Fatal("expected there to be no volumes")
  48. }
  49. }
  50. func TestInitializeWithVolumes(t *testing.T) {
  51. rootDir, err := ioutil.TempDir("", "local-volume-test")
  52. if err != nil {
  53. t.Fatal(err)
  54. }
  55. defer os.RemoveAll(rootDir)
  56. r, err := New(rootDir, 0, 0)
  57. if err != nil {
  58. t.Fatal(err)
  59. }
  60. vol, err := r.Create("testing", nil)
  61. if err != nil {
  62. t.Fatal(err)
  63. }
  64. r, err = New(rootDir, 0, 0)
  65. if err != nil {
  66. t.Fatal(err)
  67. }
  68. v, err := r.Get(vol.Name())
  69. if err != nil {
  70. t.Fatal(err)
  71. }
  72. if v.Path() != vol.Path() {
  73. t.Fatal("expected to re-initialize root with existing volumes")
  74. }
  75. }
  76. func TestCreate(t *testing.T) {
  77. rootDir, err := ioutil.TempDir("", "local-volume-test")
  78. if err != nil {
  79. t.Fatal(err)
  80. }
  81. defer os.RemoveAll(rootDir)
  82. r, err := New(rootDir, 0, 0)
  83. if err != nil {
  84. t.Fatal(err)
  85. }
  86. cases := map[string]bool{
  87. "name": true,
  88. "name-with-dash": true,
  89. "name_with_underscore": true,
  90. "name/with/slash": false,
  91. "name/with/../../slash": false,
  92. "./name": false,
  93. "../name": false,
  94. "./": false,
  95. "../": false,
  96. "~": false,
  97. ".": false,
  98. "..": false,
  99. "...": false,
  100. }
  101. for name, success := range cases {
  102. v, err := r.Create(name, nil)
  103. if success {
  104. if err != nil {
  105. t.Fatal(err)
  106. }
  107. if v.Name() != name {
  108. t.Fatalf("Expected volume with name %s, got %s", name, v.Name())
  109. }
  110. } else {
  111. if err == nil {
  112. t.Fatalf("Expected error creating volume with name %s, got nil", name)
  113. }
  114. }
  115. }
  116. }
  117. func TestValidateName(t *testing.T) {
  118. r := &Root{}
  119. names := map[string]bool{
  120. "/testvol": false,
  121. "thing.d": true,
  122. "hello-world": true,
  123. "./hello": false,
  124. ".hello": false,
  125. }
  126. for vol, expected := range names {
  127. err := r.validateName(vol)
  128. if expected && err != nil {
  129. t.Fatalf("expected %s to be valid got %v", vol, err)
  130. }
  131. if !expected && err == nil {
  132. t.Fatalf("expected %s to be invalid", vol)
  133. }
  134. }
  135. }
  136. func TestCreateWithOpts(t *testing.T) {
  137. if runtime.GOOS == "windows" {
  138. t.Skip()
  139. }
  140. rootDir, err := ioutil.TempDir("", "local-volume-test")
  141. if err != nil {
  142. t.Fatal(err)
  143. }
  144. defer os.RemoveAll(rootDir)
  145. r, err := New(rootDir, 0, 0)
  146. if err != nil {
  147. t.Fatal(err)
  148. }
  149. if _, err := r.Create("test", map[string]string{"invalidopt": "notsupported"}); err == nil {
  150. t.Fatal("expected invalid opt to cause error")
  151. }
  152. vol, err := r.Create("test", map[string]string{"device": "tmpfs", "type": "tmpfs", "o": "size=1m,uid=1000"})
  153. if err != nil {
  154. t.Fatal(err)
  155. }
  156. v := vol.(*localVolume)
  157. dir, err := v.Mount("1234")
  158. if err != nil {
  159. t.Fatal(err)
  160. }
  161. defer func() {
  162. if err := v.Unmount("1234"); err != nil {
  163. t.Fatal(err)
  164. }
  165. }()
  166. mountInfos, err := mount.GetMounts()
  167. if err != nil {
  168. t.Fatal(err)
  169. }
  170. var found bool
  171. for _, info := range mountInfos {
  172. if info.Mountpoint == dir {
  173. found = true
  174. if info.Fstype != "tmpfs" {
  175. t.Fatalf("expected tmpfs mount, got %q", info.Fstype)
  176. }
  177. if info.Source != "tmpfs" {
  178. t.Fatalf("expected tmpfs mount, got %q", info.Source)
  179. }
  180. if !strings.Contains(info.VfsOpts, "uid=1000") {
  181. t.Fatalf("expected mount info to have uid=1000: %q", info.VfsOpts)
  182. }
  183. if !strings.Contains(info.VfsOpts, "size=1024k") {
  184. t.Fatalf("expected mount info to have size=1024k: %q", info.VfsOpts)
  185. }
  186. break
  187. }
  188. }
  189. if !found {
  190. t.Fatal("mount not found")
  191. }
  192. if v.active.count != 1 {
  193. t.Fatalf("Expected active mount count to be 1, got %d", v.active.count)
  194. }
  195. // test double mount
  196. if _, err := v.Mount("1234"); err != nil {
  197. t.Fatal(err)
  198. }
  199. if v.active.count != 2 {
  200. t.Fatalf("Expected active mount count to be 2, got %d", v.active.count)
  201. }
  202. if err := v.Unmount("1234"); err != nil {
  203. t.Fatal(err)
  204. }
  205. if v.active.count != 1 {
  206. t.Fatalf("Expected active mount count to be 1, got %d", v.active.count)
  207. }
  208. mounted, err := mount.Mounted(v.path)
  209. if err != nil {
  210. t.Fatal(err)
  211. }
  212. if !mounted {
  213. t.Fatal("expected mount to still be active")
  214. }
  215. r, err = New(rootDir, 0, 0)
  216. if err != nil {
  217. t.Fatal(err)
  218. }
  219. v2, exists := r.volumes["test"]
  220. if !exists {
  221. t.Fatal("missing volume on restart")
  222. }
  223. if !reflect.DeepEqual(v.opts, v2.opts) {
  224. t.Fatal("missing volume options on restart")
  225. }
  226. }