server_test.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588
  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("inspect", container1.Name, "container")
  67. baseContainer, _ := job.Stdout.AddEnv()
  68. if err := job.Run(); err != nil {
  69. t.Error(err)
  70. }
  71. job = eng.Job("inspect", container2.Name, "container")
  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. // FIXE: 'insert' is deprecated and should be removed in a future version.
  323. func TestImageInsert(t *testing.T) {
  324. eng := NewTestEngine(t)
  325. defer mkDaemonFromEngine(eng, t).Nuke()
  326. srv := mkServerFromEngine(eng, t)
  327. // bad image name fails
  328. if err := srv.Eng.Job("insert", "foo", "https://www.docker.io/static/img/docker-top-logo.png", "/foo").Run(); err == nil {
  329. t.Fatal("expected an error and got none")
  330. }
  331. // bad url fails
  332. if err := srv.Eng.Job("insert", unitTestImageID, "http://bad_host_name_that_will_totally_fail.com/", "/foo").Run(); err == nil {
  333. t.Fatal("expected an error and got none")
  334. }
  335. // success returns nil
  336. if err := srv.Eng.Job("insert", unitTestImageID, "https://www.docker.io/static/img/docker-top-logo.png", "/foo").Run(); err != nil {
  337. t.Fatalf("expected no error, but got %v", err)
  338. }
  339. }
  340. func TestListContainers(t *testing.T) {
  341. eng := NewTestEngine(t)
  342. srv := mkServerFromEngine(eng, t)
  343. defer mkDaemonFromEngine(eng, t).Nuke()
  344. config := runconfig.Config{
  345. Image: unitTestImageID,
  346. Cmd: []string{"/bin/sh", "-c", "cat"},
  347. OpenStdin: true,
  348. }
  349. firstID := createTestContainer(eng, &config, t)
  350. secondID := createTestContainer(eng, &config, t)
  351. thirdID := createTestContainer(eng, &config, t)
  352. fourthID := createTestContainer(eng, &config, t)
  353. defer func() {
  354. containerKill(eng, firstID, t)
  355. containerKill(eng, secondID, t)
  356. containerKill(eng, fourthID, t)
  357. containerWait(eng, firstID, t)
  358. containerWait(eng, secondID, t)
  359. containerWait(eng, fourthID, t)
  360. }()
  361. startContainer(eng, firstID, t)
  362. startContainer(eng, secondID, t)
  363. startContainer(eng, fourthID, t)
  364. // all
  365. if !assertContainerList(srv, true, -1, "", "", []string{fourthID, thirdID, secondID, firstID}) {
  366. t.Error("Container list is not in the correct order")
  367. }
  368. // running
  369. if !assertContainerList(srv, false, -1, "", "", []string{fourthID, secondID, firstID}) {
  370. t.Error("Container list is not in the correct order")
  371. }
  372. // from here 'all' flag is ignored
  373. // limit
  374. expected := []string{fourthID, thirdID}
  375. if !assertContainerList(srv, true, 2, "", "", expected) ||
  376. !assertContainerList(srv, false, 2, "", "", expected) {
  377. t.Error("Container list is not in the correct order")
  378. }
  379. // since
  380. expected = []string{fourthID, thirdID, secondID}
  381. if !assertContainerList(srv, true, -1, firstID, "", expected) ||
  382. !assertContainerList(srv, false, -1, firstID, "", expected) {
  383. t.Error("Container list is not in the correct order")
  384. }
  385. // before
  386. expected = []string{secondID, firstID}
  387. if !assertContainerList(srv, true, -1, "", thirdID, expected) ||
  388. !assertContainerList(srv, false, -1, "", thirdID, expected) {
  389. t.Error("Container list is not in the correct order")
  390. }
  391. // since & before
  392. expected = []string{thirdID, secondID}
  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. // since & limit
  398. expected = []string{fourthID, thirdID}
  399. if !assertContainerList(srv, true, 2, firstID, "", expected) ||
  400. !assertContainerList(srv, false, 2, firstID, "", expected) {
  401. t.Error("Container list is not in the correct order")
  402. }
  403. // before & limit
  404. expected = []string{thirdID}
  405. if !assertContainerList(srv, true, 1, "", fourthID, expected) ||
  406. !assertContainerList(srv, false, 1, "", fourthID, expected) {
  407. t.Error("Container list is not in the correct order")
  408. }
  409. // since & before & limit
  410. expected = []string{thirdID}
  411. if !assertContainerList(srv, true, 1, firstID, fourthID, expected) ||
  412. !assertContainerList(srv, false, 1, firstID, fourthID, expected) {
  413. t.Error("Container list is not in the correct order")
  414. }
  415. }
  416. func assertContainerList(srv *server.Server, all bool, limit int, since, before string, expected []string) bool {
  417. job := srv.Eng.Job("containers")
  418. job.SetenvBool("all", all)
  419. job.SetenvInt("limit", limit)
  420. job.Setenv("since", since)
  421. job.Setenv("before", before)
  422. outs, err := job.Stdout.AddListTable()
  423. if err != nil {
  424. return false
  425. }
  426. if err := job.Run(); err != nil {
  427. return false
  428. }
  429. if len(outs.Data) != len(expected) {
  430. return false
  431. }
  432. for i := 0; i < len(outs.Data); i++ {
  433. if outs.Data[i].Get("Id") != expected[i] {
  434. return false
  435. }
  436. }
  437. return true
  438. }
  439. // Regression test for being able to untag an image with an existing
  440. // container
  441. func TestDeleteTagWithExistingContainers(t *testing.T) {
  442. eng := NewTestEngine(t)
  443. defer nuke(mkDaemonFromEngine(eng, t))
  444. srv := mkServerFromEngine(eng, t)
  445. // Tag the image
  446. if err := eng.Job("tag", unitTestImageID, "utest", "tag1").Run(); err != nil {
  447. t.Fatal(err)
  448. }
  449. // Create a container from the image
  450. config, _, _, err := runconfig.Parse([]string{unitTestImageID, "echo test"}, nil)
  451. if err != nil {
  452. t.Fatal(err)
  453. }
  454. id := createNamedTestContainer(eng, config, t, "testingtags")
  455. if id == "" {
  456. t.Fatal("No id returned")
  457. }
  458. job := srv.Eng.Job("containers")
  459. job.SetenvBool("all", true)
  460. outs, err := job.Stdout.AddListTable()
  461. if err != nil {
  462. t.Fatal(err)
  463. }
  464. if err := job.Run(); err != nil {
  465. t.Fatal(err)
  466. }
  467. if len(outs.Data) != 1 {
  468. t.Fatalf("Expected 1 container got %d", len(outs.Data))
  469. }
  470. // Try to remove the tag
  471. imgs := engine.NewTable("", 0)
  472. if err := srv.DeleteImage("utest:tag1", imgs, true, false, false); err != nil {
  473. t.Fatal(err)
  474. }
  475. if len(imgs.Data) != 1 {
  476. t.Fatalf("Should only have deleted one untag %d", len(imgs.Data))
  477. }
  478. if untag := imgs.Data[0].Get("Untagged"); untag != "utest:tag1" {
  479. t.Fatalf("Expected %s got %s", unitTestImageID, untag)
  480. }
  481. }