server_test.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566
  1. package docker
  2. import (
  3. "bytes"
  4. "strings"
  5. "testing"
  6. "time"
  7. "github.com/dotcloud/docker/engine"
  8. "github.com/dotcloud/docker/runconfig"
  9. "github.com/dotcloud/docker/server"
  10. )
  11. func TestCreateNumberHostname(t *testing.T) {
  12. eng := NewTestEngine(t)
  13. defer mkDaemonFromEngine(eng, t).Nuke()
  14. config, _, _, err := runconfig.Parse([]string{"-h", "web.0", unitTestImageID, "echo test"}, nil)
  15. if err != nil {
  16. t.Fatal(err)
  17. }
  18. createTestContainer(eng, config, t)
  19. }
  20. func TestCreateNumberUsername(t *testing.T) {
  21. eng := NewTestEngine(t)
  22. defer mkDaemonFromEngine(eng, t).Nuke()
  23. config, _, _, err := runconfig.Parse([]string{"-u", "1002", unitTestImageID, "echo test"}, nil)
  24. if err != nil {
  25. t.Fatal(err)
  26. }
  27. createTestContainer(eng, config, t)
  28. }
  29. func TestCommit(t *testing.T) {
  30. eng := NewTestEngine(t)
  31. defer mkDaemonFromEngine(eng, t).Nuke()
  32. config, _, _, err := runconfig.Parse([]string{unitTestImageID, "/bin/cat"}, nil)
  33. if err != nil {
  34. t.Fatal(err)
  35. }
  36. id := createTestContainer(eng, config, t)
  37. job := eng.Job("commit", id)
  38. job.Setenv("repo", "testrepo")
  39. job.Setenv("tag", "testtag")
  40. job.SetenvJson("config", config)
  41. if err := job.Run(); err != nil {
  42. t.Fatal(err)
  43. }
  44. }
  45. func TestMergeConfigOnCommit(t *testing.T) {
  46. eng := NewTestEngine(t)
  47. runtime := mkDaemonFromEngine(eng, t)
  48. defer runtime.Nuke()
  49. container1, _, _ := mkContainer(runtime, []string{"-e", "FOO=bar", unitTestImageID, "echo test > /tmp/foo"}, t)
  50. defer runtime.Destroy(container1)
  51. config, _, _, err := runconfig.Parse([]string{container1.ID, "cat /tmp/foo"}, nil)
  52. if err != nil {
  53. t.Error(err)
  54. }
  55. job := eng.Job("commit", container1.ID)
  56. job.Setenv("repo", "testrepo")
  57. job.Setenv("tag", "testtag")
  58. job.SetenvJson("config", config)
  59. var outputBuffer = bytes.NewBuffer(nil)
  60. job.Stdout.Add(outputBuffer)
  61. if err := job.Run(); err != nil {
  62. t.Error(err)
  63. }
  64. container2, _, _ := mkContainer(runtime, []string{engine.Tail(outputBuffer, 1)}, t)
  65. defer runtime.Destroy(container2)
  66. job = eng.Job("container_inspect", container1.Name)
  67. baseContainer, _ := job.Stdout.AddEnv()
  68. if err := job.Run(); err != nil {
  69. t.Error(err)
  70. }
  71. job = eng.Job("container_inspect", container2.Name)
  72. commitContainer, _ := job.Stdout.AddEnv()
  73. if err := job.Run(); err != nil {
  74. t.Error(err)
  75. }
  76. baseConfig := baseContainer.GetSubEnv("Config")
  77. commitConfig := commitContainer.GetSubEnv("Config")
  78. if commitConfig.Get("Env") != baseConfig.Get("Env") {
  79. t.Fatalf("Env config in committed container should be %v, was %v",
  80. baseConfig.Get("Env"), commitConfig.Get("Env"))
  81. }
  82. if baseConfig.Get("Cmd") != "[\"echo test \\u003e /tmp/foo\"]" {
  83. t.Fatalf("Cmd in base container should be [\"echo test \\u003e /tmp/foo\"], was %s",
  84. baseConfig.Get("Cmd"))
  85. }
  86. if commitConfig.Get("Cmd") != "[\"cat /tmp/foo\"]" {
  87. t.Fatalf("Cmd in committed container should be [\"cat /tmp/foo\"], was %s",
  88. commitConfig.Get("Cmd"))
  89. }
  90. }
  91. func TestRestartKillWait(t *testing.T) {
  92. eng := NewTestEngine(t)
  93. srv := mkServerFromEngine(eng, t)
  94. runtime := mkDaemonFromEngine(eng, t)
  95. defer runtime.Nuke()
  96. config, hostConfig, _, err := runconfig.Parse([]string{"-i", unitTestImageID, "/bin/cat"}, nil)
  97. if err != nil {
  98. t.Fatal(err)
  99. }
  100. id := createTestContainer(eng, config, t)
  101. job := eng.Job("containers")
  102. job.SetenvBool("all", true)
  103. outs, err := job.Stdout.AddListTable()
  104. if err != nil {
  105. t.Fatal(err)
  106. }
  107. if err := job.Run(); err != nil {
  108. t.Fatal(err)
  109. }
  110. if len(outs.Data) != 1 {
  111. t.Errorf("Expected 1 container, %v found", len(outs.Data))
  112. }
  113. job = eng.Job("start", id)
  114. if err := job.ImportEnv(hostConfig); err != nil {
  115. t.Fatal(err)
  116. }
  117. if err := job.Run(); err != nil {
  118. t.Fatal(err)
  119. }
  120. job = eng.Job("kill", id)
  121. if err := job.Run(); err != nil {
  122. t.Fatal(err)
  123. }
  124. eng = newTestEngine(t, false, runtime.Config().Root)
  125. srv = mkServerFromEngine(eng, t)
  126. job = srv.Eng.Job("containers")
  127. job.SetenvBool("all", true)
  128. outs, err = job.Stdout.AddListTable()
  129. if err != nil {
  130. t.Fatal(err)
  131. }
  132. if err := job.Run(); err != nil {
  133. t.Fatal(err)
  134. }
  135. if len(outs.Data) != 1 {
  136. t.Errorf("Expected 1 container, %v found", len(outs.Data))
  137. }
  138. setTimeout(t, "Waiting on stopped container timedout", 5*time.Second, func() {
  139. job = srv.Eng.Job("wait", outs.Data[0].Get("Id"))
  140. if err := job.Run(); err != nil {
  141. t.Fatal(err)
  142. }
  143. })
  144. }
  145. func TestCreateStartRestartStopStartKillRm(t *testing.T) {
  146. eng := NewTestEngine(t)
  147. srv := mkServerFromEngine(eng, t)
  148. defer mkDaemonFromEngine(eng, t).Nuke()
  149. config, hostConfig, _, err := runconfig.Parse([]string{"-i", unitTestImageID, "/bin/cat"}, nil)
  150. if err != nil {
  151. t.Fatal(err)
  152. }
  153. id := createTestContainer(eng, config, t)
  154. job := srv.Eng.Job("containers")
  155. job.SetenvBool("all", true)
  156. outs, err := job.Stdout.AddListTable()
  157. if err != nil {
  158. t.Fatal(err)
  159. }
  160. if err := job.Run(); err != nil {
  161. t.Fatal(err)
  162. }
  163. if len(outs.Data) != 1 {
  164. t.Errorf("Expected 1 container, %v found", len(outs.Data))
  165. }
  166. job = eng.Job("start", id)
  167. if err := job.ImportEnv(hostConfig); err != nil {
  168. t.Fatal(err)
  169. }
  170. if err := job.Run(); err != nil {
  171. t.Fatal(err)
  172. }
  173. job = eng.Job("restart", id)
  174. job.SetenvInt("t", 15)
  175. if err := job.Run(); err != nil {
  176. t.Fatal(err)
  177. }
  178. job = eng.Job("stop", id)
  179. job.SetenvInt("t", 15)
  180. if err := job.Run(); err != nil {
  181. t.Fatal(err)
  182. }
  183. job = eng.Job("start", id)
  184. if err := job.ImportEnv(hostConfig); err != nil {
  185. t.Fatal(err)
  186. }
  187. if err := job.Run(); err != nil {
  188. t.Fatal(err)
  189. }
  190. if err := eng.Job("kill", id).Run(); err != nil {
  191. t.Fatal(err)
  192. }
  193. // FIXME: this failed once with a race condition ("Unable to remove filesystem for xxx: directory not empty")
  194. job = eng.Job("container_delete", id)
  195. job.SetenvBool("removeVolume", true)
  196. if err := job.Run(); err != nil {
  197. t.Fatal(err)
  198. }
  199. job = srv.Eng.Job("containers")
  200. job.SetenvBool("all", true)
  201. outs, err = job.Stdout.AddListTable()
  202. if err != nil {
  203. t.Fatal(err)
  204. }
  205. if err := job.Run(); err != nil {
  206. t.Fatal(err)
  207. }
  208. if len(outs.Data) != 0 {
  209. t.Errorf("Expected 0 container, %v found", len(outs.Data))
  210. }
  211. }
  212. func TestRunWithTooLowMemoryLimit(t *testing.T) {
  213. eng := NewTestEngine(t)
  214. defer mkDaemonFromEngine(eng, t).Nuke()
  215. // Try to create a container with a memory limit of 1 byte less than the minimum allowed limit.
  216. job := eng.Job("create")
  217. job.Setenv("Image", unitTestImageID)
  218. job.Setenv("Memory", "524287")
  219. job.Setenv("CpuShares", "1000")
  220. job.SetenvList("Cmd", []string{"/bin/cat"})
  221. if err := job.Run(); err == nil {
  222. t.Errorf("Memory limit is smaller than the allowed limit. Container creation should've failed!")
  223. }
  224. }
  225. func TestRmi(t *testing.T) {
  226. eng := NewTestEngine(t)
  227. srv := mkServerFromEngine(eng, t)
  228. defer mkDaemonFromEngine(eng, t).Nuke()
  229. initialImages := getAllImages(eng, t)
  230. config, hostConfig, _, err := runconfig.Parse([]string{unitTestImageID, "echo", "test"}, nil)
  231. if err != nil {
  232. t.Fatal(err)
  233. }
  234. containerID := createTestContainer(eng, config, t)
  235. //To remove
  236. job := eng.Job("start", containerID)
  237. if err := job.ImportEnv(hostConfig); err != nil {
  238. t.Fatal(err)
  239. }
  240. if err := job.Run(); err != nil {
  241. t.Fatal(err)
  242. }
  243. if err := eng.Job("wait", containerID).Run(); err != nil {
  244. t.Fatal(err)
  245. }
  246. job = eng.Job("commit", containerID)
  247. job.Setenv("repo", "test")
  248. var outputBuffer = bytes.NewBuffer(nil)
  249. job.Stdout.Add(outputBuffer)
  250. if err := job.Run(); err != nil {
  251. t.Fatal(err)
  252. }
  253. if err := eng.Job("tag", engine.Tail(outputBuffer, 1), "test", "0.1").Run(); err != nil {
  254. t.Fatal(err)
  255. }
  256. containerID = createTestContainer(eng, config, t)
  257. //To remove
  258. job = eng.Job("start", containerID)
  259. if err := job.ImportEnv(hostConfig); err != nil {
  260. t.Fatal(err)
  261. }
  262. if err := job.Run(); err != nil {
  263. t.Fatal(err)
  264. }
  265. if err := eng.Job("wait", containerID).Run(); err != nil {
  266. t.Fatal(err)
  267. }
  268. job = eng.Job("commit", containerID)
  269. job.Setenv("repo", "test")
  270. if err := job.Run(); err != nil {
  271. t.Fatal(err)
  272. }
  273. images := getAllImages(eng, t)
  274. if images.Len()-initialImages.Len() != 2 {
  275. t.Fatalf("Expected 2 new images, found %d.", images.Len()-initialImages.Len())
  276. }
  277. if err = srv.DeleteImage(engine.Tail(outputBuffer, 1), engine.NewTable("", 0), true, false, false); err != nil {
  278. t.Fatal(err)
  279. }
  280. images = getAllImages(eng, t)
  281. if images.Len()-initialImages.Len() != 1 {
  282. t.Fatalf("Expected 1 new image, found %d.", images.Len()-initialImages.Len())
  283. }
  284. for _, image := range images.Data {
  285. if strings.Contains(unitTestImageID, image.Get("Id")) {
  286. continue
  287. }
  288. if image.GetList("RepoTags")[0] == "<none>:<none>" {
  289. t.Fatalf("Expected tagged image, got untagged one.")
  290. }
  291. }
  292. }
  293. func TestImagesFilter(t *testing.T) {
  294. eng := NewTestEngine(t)
  295. defer nuke(mkDaemonFromEngine(eng, t))
  296. if err := eng.Job("tag", unitTestImageName, "utest", "tag1").Run(); err != nil {
  297. t.Fatal(err)
  298. }
  299. if err := eng.Job("tag", unitTestImageName, "utest/docker", "tag2").Run(); err != nil {
  300. t.Fatal(err)
  301. }
  302. if err := eng.Job("tag", unitTestImageName, "utest:5000/docker", "tag3").Run(); err != nil {
  303. t.Fatal(err)
  304. }
  305. images := getImages(eng, t, false, "utest*/*")
  306. if len(images.Data[0].GetList("RepoTags")) != 2 {
  307. t.Fatal("incorrect number of matches returned")
  308. }
  309. images = getImages(eng, t, false, "utest")
  310. if len(images.Data[0].GetList("RepoTags")) != 1 {
  311. t.Fatal("incorrect number of matches returned")
  312. }
  313. images = getImages(eng, t, false, "utest*")
  314. if len(images.Data[0].GetList("RepoTags")) != 1 {
  315. t.Fatal("incorrect number of matches returned")
  316. }
  317. images = getImages(eng, t, false, "*5000*/*")
  318. if len(images.Data[0].GetList("RepoTags")) != 1 {
  319. t.Fatal("incorrect number of matches returned")
  320. }
  321. }
  322. func TestListContainers(t *testing.T) {
  323. eng := NewTestEngine(t)
  324. srv := mkServerFromEngine(eng, t)
  325. defer mkDaemonFromEngine(eng, t).Nuke()
  326. config := runconfig.Config{
  327. Image: unitTestImageID,
  328. Cmd: []string{"/bin/sh", "-c", "cat"},
  329. OpenStdin: true,
  330. }
  331. firstID := createTestContainer(eng, &config, t)
  332. secondID := createTestContainer(eng, &config, t)
  333. thirdID := createTestContainer(eng, &config, t)
  334. fourthID := createTestContainer(eng, &config, t)
  335. defer func() {
  336. containerKill(eng, firstID, t)
  337. containerKill(eng, secondID, t)
  338. containerKill(eng, fourthID, t)
  339. containerWait(eng, firstID, t)
  340. containerWait(eng, secondID, t)
  341. containerWait(eng, fourthID, t)
  342. }()
  343. startContainer(eng, firstID, t)
  344. startContainer(eng, secondID, t)
  345. startContainer(eng, fourthID, t)
  346. // all
  347. if !assertContainerList(srv, true, -1, "", "", []string{fourthID, thirdID, secondID, firstID}) {
  348. t.Error("Container list is not in the correct order")
  349. }
  350. // running
  351. if !assertContainerList(srv, false, -1, "", "", []string{fourthID, secondID, firstID}) {
  352. t.Error("Container list is not in the correct order")
  353. }
  354. // from here 'all' flag is ignored
  355. // limit
  356. expected := []string{fourthID, thirdID}
  357. if !assertContainerList(srv, true, 2, "", "", expected) ||
  358. !assertContainerList(srv, false, 2, "", "", expected) {
  359. t.Error("Container list is not in the correct order")
  360. }
  361. // since
  362. expected = []string{fourthID, thirdID, secondID}
  363. if !assertContainerList(srv, true, -1, firstID, "", expected) ||
  364. !assertContainerList(srv, false, -1, firstID, "", expected) {
  365. t.Error("Container list is not in the correct order")
  366. }
  367. // before
  368. expected = []string{secondID, firstID}
  369. if !assertContainerList(srv, true, -1, "", thirdID, expected) ||
  370. !assertContainerList(srv, false, -1, "", thirdID, expected) {
  371. t.Error("Container list is not in the correct order")
  372. }
  373. // since & before
  374. expected = []string{thirdID, secondID}
  375. if !assertContainerList(srv, true, -1, firstID, fourthID, expected) ||
  376. !assertContainerList(srv, false, -1, firstID, fourthID, expected) {
  377. t.Error("Container list is not in the correct order")
  378. }
  379. // since & limit
  380. expected = []string{fourthID, thirdID}
  381. if !assertContainerList(srv, true, 2, firstID, "", expected) ||
  382. !assertContainerList(srv, false, 2, firstID, "", expected) {
  383. t.Error("Container list is not in the correct order")
  384. }
  385. // before & limit
  386. expected = []string{thirdID}
  387. if !assertContainerList(srv, true, 1, "", fourthID, expected) ||
  388. !assertContainerList(srv, false, 1, "", fourthID, expected) {
  389. t.Error("Container list is not in the correct order")
  390. }
  391. // since & before & limit
  392. expected = []string{thirdID}
  393. if !assertContainerList(srv, true, 1, firstID, fourthID, expected) ||
  394. !assertContainerList(srv, false, 1, firstID, fourthID, expected) {
  395. t.Error("Container list is not in the correct order")
  396. }
  397. }
  398. func assertContainerList(srv *server.Server, all bool, limit int, since, before string, expected []string) bool {
  399. job := srv.Eng.Job("containers")
  400. job.SetenvBool("all", all)
  401. job.SetenvInt("limit", limit)
  402. job.Setenv("since", since)
  403. job.Setenv("before", before)
  404. outs, err := job.Stdout.AddListTable()
  405. if err != nil {
  406. return false
  407. }
  408. if err := job.Run(); err != nil {
  409. return false
  410. }
  411. if len(outs.Data) != len(expected) {
  412. return false
  413. }
  414. for i := 0; i < len(outs.Data); i++ {
  415. if outs.Data[i].Get("Id") != expected[i] {
  416. return false
  417. }
  418. }
  419. return true
  420. }
  421. // Regression test for being able to untag an image with an existing
  422. // container
  423. func TestDeleteTagWithExistingContainers(t *testing.T) {
  424. eng := NewTestEngine(t)
  425. defer nuke(mkDaemonFromEngine(eng, t))
  426. srv := mkServerFromEngine(eng, t)
  427. // Tag the image
  428. if err := eng.Job("tag", unitTestImageID, "utest", "tag1").Run(); err != nil {
  429. t.Fatal(err)
  430. }
  431. // Create a container from the image
  432. config, _, _, err := runconfig.Parse([]string{unitTestImageID, "echo test"}, nil)
  433. if err != nil {
  434. t.Fatal(err)
  435. }
  436. id := createNamedTestContainer(eng, config, t, "testingtags")
  437. if id == "" {
  438. t.Fatal("No id returned")
  439. }
  440. job := srv.Eng.Job("containers")
  441. job.SetenvBool("all", true)
  442. outs, err := job.Stdout.AddListTable()
  443. if err != nil {
  444. t.Fatal(err)
  445. }
  446. if err := job.Run(); err != nil {
  447. t.Fatal(err)
  448. }
  449. if len(outs.Data) != 1 {
  450. t.Fatalf("Expected 1 container got %d", len(outs.Data))
  451. }
  452. // Try to remove the tag
  453. imgs := engine.NewTable("", 0)
  454. if err := srv.DeleteImage("utest:tag1", imgs, true, false, false); err != nil {
  455. t.Fatal(err)
  456. }
  457. if len(imgs.Data) != 1 {
  458. t.Fatalf("Should only have deleted one untag %d", len(imgs.Data))
  459. }
  460. if untag := imgs.Data[0].Get("Untagged"); untag != "utest:tag1" {
  461. t.Fatalf("Expected %s got %s", unitTestImageID, untag)
  462. }
  463. }