local_test.go 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344
  1. package local
  2. import (
  3. "io/ioutil"
  4. "os"
  5. "path/filepath"
  6. "reflect"
  7. "runtime"
  8. "strings"
  9. "testing"
  10. "github.com/docker/docker/pkg/mount"
  11. )
  12. func TestGetAddress(t *testing.T) {
  13. cases := map[string]string{
  14. "addr=11.11.11.1": "11.11.11.1",
  15. " ": "",
  16. "addr=": "",
  17. "addr=2001:db8::68": "2001:db8::68",
  18. }
  19. for name, success := range cases {
  20. v := getAddress(name)
  21. if v != success {
  22. t.Errorf("Test case failed for %s actual: %s expected : %s", name, v, success)
  23. }
  24. }
  25. }
  26. func TestRemove(t *testing.T) {
  27. // TODO Windows: Investigate why this test fails on Windows under CI
  28. // but passes locally.
  29. if runtime.GOOS == "windows" {
  30. t.Skip("Test failing on Windows CI")
  31. }
  32. rootDir, err := ioutil.TempDir("", "local-volume-test")
  33. if err != nil {
  34. t.Fatal(err)
  35. }
  36. defer os.RemoveAll(rootDir)
  37. r, err := New(rootDir, 0, 0)
  38. if err != nil {
  39. t.Fatal(err)
  40. }
  41. vol, err := r.Create("testing", nil)
  42. if err != nil {
  43. t.Fatal(err)
  44. }
  45. if err := r.Remove(vol); err != nil {
  46. t.Fatal(err)
  47. }
  48. vol, err = r.Create("testing2", nil)
  49. if err != nil {
  50. t.Fatal(err)
  51. }
  52. if err := os.RemoveAll(vol.Path()); err != nil {
  53. t.Fatal(err)
  54. }
  55. if err := r.Remove(vol); err != nil {
  56. t.Fatal(err)
  57. }
  58. if _, err := os.Stat(vol.Path()); err != nil && !os.IsNotExist(err) {
  59. t.Fatal("volume dir not removed")
  60. }
  61. if l, _ := r.List(); len(l) != 0 {
  62. t.Fatal("expected there to be no volumes")
  63. }
  64. }
  65. func TestInitializeWithVolumes(t *testing.T) {
  66. rootDir, err := ioutil.TempDir("", "local-volume-test")
  67. if err != nil {
  68. t.Fatal(err)
  69. }
  70. defer os.RemoveAll(rootDir)
  71. r, err := New(rootDir, 0, 0)
  72. if err != nil {
  73. t.Fatal(err)
  74. }
  75. vol, err := r.Create("testing", nil)
  76. if err != nil {
  77. t.Fatal(err)
  78. }
  79. r, err = New(rootDir, 0, 0)
  80. if err != nil {
  81. t.Fatal(err)
  82. }
  83. v, err := r.Get(vol.Name())
  84. if err != nil {
  85. t.Fatal(err)
  86. }
  87. if v.Path() != vol.Path() {
  88. t.Fatal("expected to re-initialize root with existing volumes")
  89. }
  90. }
  91. func TestCreate(t *testing.T) {
  92. rootDir, err := ioutil.TempDir("", "local-volume-test")
  93. if err != nil {
  94. t.Fatal(err)
  95. }
  96. defer os.RemoveAll(rootDir)
  97. r, err := New(rootDir, 0, 0)
  98. if err != nil {
  99. t.Fatal(err)
  100. }
  101. cases := map[string]bool{
  102. "name": true,
  103. "name-with-dash": true,
  104. "name_with_underscore": true,
  105. "name/with/slash": false,
  106. "name/with/../../slash": false,
  107. "./name": false,
  108. "../name": false,
  109. "./": false,
  110. "../": false,
  111. "~": false,
  112. ".": false,
  113. "..": false,
  114. "...": false,
  115. }
  116. for name, success := range cases {
  117. v, err := r.Create(name, nil)
  118. if success {
  119. if err != nil {
  120. t.Fatal(err)
  121. }
  122. if v.Name() != name {
  123. t.Fatalf("Expected volume with name %s, got %s", name, v.Name())
  124. }
  125. } else {
  126. if err == nil {
  127. t.Fatalf("Expected error creating volume with name %s, got nil", name)
  128. }
  129. }
  130. }
  131. r, err = New(rootDir, 0, 0)
  132. if err != nil {
  133. t.Fatal(err)
  134. }
  135. }
  136. func TestValidateName(t *testing.T) {
  137. r := &Root{}
  138. names := map[string]bool{
  139. "x": false,
  140. "/testvol": false,
  141. "thing.d": true,
  142. "hello-world": true,
  143. "./hello": false,
  144. ".hello": false,
  145. }
  146. for vol, expected := range names {
  147. err := r.validateName(vol)
  148. if expected && err != nil {
  149. t.Fatalf("expected %s to be valid got %v", vol, err)
  150. }
  151. if !expected && err == nil {
  152. t.Fatalf("expected %s to be invalid", vol)
  153. }
  154. }
  155. }
  156. func TestCreateWithOpts(t *testing.T) {
  157. if runtime.GOOS == "windows" || runtime.GOOS == "solaris" {
  158. t.Skip()
  159. }
  160. rootDir, err := ioutil.TempDir("", "local-volume-test")
  161. if err != nil {
  162. t.Fatal(err)
  163. }
  164. defer os.RemoveAll(rootDir)
  165. r, err := New(rootDir, 0, 0)
  166. if err != nil {
  167. t.Fatal(err)
  168. }
  169. if _, err := r.Create("test", map[string]string{"invalidopt": "notsupported"}); err == nil {
  170. t.Fatal("expected invalid opt to cause error")
  171. }
  172. vol, err := r.Create("test", map[string]string{"device": "tmpfs", "type": "tmpfs", "o": "size=1m,uid=1000"})
  173. if err != nil {
  174. t.Fatal(err)
  175. }
  176. v := vol.(*localVolume)
  177. dir, err := v.Mount("1234")
  178. if err != nil {
  179. t.Fatal(err)
  180. }
  181. defer func() {
  182. if err := v.Unmount("1234"); err != nil {
  183. t.Fatal(err)
  184. }
  185. }()
  186. mountInfos, err := mount.GetMounts()
  187. if err != nil {
  188. t.Fatal(err)
  189. }
  190. var found bool
  191. for _, info := range mountInfos {
  192. if info.Mountpoint == dir {
  193. found = true
  194. if info.Fstype != "tmpfs" {
  195. t.Fatalf("expected tmpfs mount, got %q", info.Fstype)
  196. }
  197. if info.Source != "tmpfs" {
  198. t.Fatalf("expected tmpfs mount, got %q", info.Source)
  199. }
  200. if !strings.Contains(info.VfsOpts, "uid=1000") {
  201. t.Fatalf("expected mount info to have uid=1000: %q", info.VfsOpts)
  202. }
  203. if !strings.Contains(info.VfsOpts, "size=1024k") {
  204. t.Fatalf("expected mount info to have size=1024k: %q", info.VfsOpts)
  205. }
  206. break
  207. }
  208. }
  209. if !found {
  210. t.Fatal("mount not found")
  211. }
  212. if v.active.count != 1 {
  213. t.Fatalf("Expected active mount count to be 1, got %d", v.active.count)
  214. }
  215. // test double mount
  216. if _, err := v.Mount("1234"); err != nil {
  217. t.Fatal(err)
  218. }
  219. if v.active.count != 2 {
  220. t.Fatalf("Expected active mount count to be 2, got %d", v.active.count)
  221. }
  222. if err := v.Unmount("1234"); err != nil {
  223. t.Fatal(err)
  224. }
  225. if v.active.count != 1 {
  226. t.Fatalf("Expected active mount count to be 1, got %d", v.active.count)
  227. }
  228. mounted, err := mount.Mounted(v.path)
  229. if err != nil {
  230. t.Fatal(err)
  231. }
  232. if !mounted {
  233. t.Fatal("expected mount to still be active")
  234. }
  235. r, err = New(rootDir, 0, 0)
  236. if err != nil {
  237. t.Fatal(err)
  238. }
  239. v2, exists := r.volumes["test"]
  240. if !exists {
  241. t.Fatal("missing volume on restart")
  242. }
  243. if !reflect.DeepEqual(v.opts, v2.opts) {
  244. t.Fatal("missing volume options on restart")
  245. }
  246. }
  247. func TestRealodNoOpts(t *testing.T) {
  248. rootDir, err := ioutil.TempDir("", "volume-test-reload-no-opts")
  249. if err != nil {
  250. t.Fatal(err)
  251. }
  252. defer os.RemoveAll(rootDir)
  253. r, err := New(rootDir, 0, 0)
  254. if err != nil {
  255. t.Fatal(err)
  256. }
  257. if _, err := r.Create("test1", nil); err != nil {
  258. t.Fatal(err)
  259. }
  260. if _, err := r.Create("test2", nil); err != nil {
  261. t.Fatal(err)
  262. }
  263. // make sure a file with `null` (.e.g. empty opts map from older daemon) is ok
  264. if err := ioutil.WriteFile(filepath.Join(rootDir, "test2"), []byte("null"), 600); err != nil {
  265. t.Fatal(err)
  266. }
  267. if _, err := r.Create("test3", nil); err != nil {
  268. t.Fatal(err)
  269. }
  270. // make sure an empty opts file doesn't break us too
  271. if err := ioutil.WriteFile(filepath.Join(rootDir, "test3"), nil, 600); err != nil {
  272. t.Fatal(err)
  273. }
  274. if _, err := r.Create("test4", map[string]string{}); err != nil {
  275. t.Fatal(err)
  276. }
  277. r, err = New(rootDir, 0, 0)
  278. if err != nil {
  279. t.Fatal(err)
  280. }
  281. for _, name := range []string{"test1", "test2", "test3", "test4"} {
  282. v, err := r.Get(name)
  283. if err != nil {
  284. t.Fatal(err)
  285. }
  286. lv, ok := v.(*localVolume)
  287. if !ok {
  288. t.Fatalf("expected *localVolume got: %v", reflect.TypeOf(v))
  289. }
  290. if lv.opts != nil {
  291. t.Fatalf("expected opts to be nil, got: %v", lv.opts)
  292. }
  293. if _, err := lv.Mount("1234"); err != nil {
  294. t.Fatal(err)
  295. }
  296. }
  297. }