custom_test.go 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. package formatter
  2. import (
  3. "reflect"
  4. "strings"
  5. "testing"
  6. "time"
  7. "github.com/docker/docker/pkg/stringid"
  8. "github.com/docker/engine-api/types"
  9. )
  10. func TestContainerPsContext(t *testing.T) {
  11. containerID := stringid.GenerateRandomID()
  12. unix := time.Now().Add(-65 * time.Second).Unix()
  13. var ctx containerContext
  14. cases := []struct {
  15. container types.Container
  16. trunc bool
  17. expValue string
  18. expHeader string
  19. call func() string
  20. }{
  21. {types.Container{ID: containerID}, true, stringid.TruncateID(containerID), containerIDHeader, ctx.ID},
  22. {types.Container{ID: containerID}, false, containerID, containerIDHeader, ctx.ID},
  23. {types.Container{Names: []string{"/foobar_baz"}}, true, "foobar_baz", namesHeader, ctx.Names},
  24. {types.Container{Image: "ubuntu"}, true, "ubuntu", imageHeader, ctx.Image},
  25. {types.Container{Image: "verylongimagename"}, true, "verylongimagename", imageHeader, ctx.Image},
  26. {types.Container{Image: "verylongimagename"}, false, "verylongimagename", imageHeader, ctx.Image},
  27. {types.Container{
  28. Image: "a5a665ff33eced1e0803148700880edab4",
  29. ImageID: "a5a665ff33eced1e0803148700880edab4269067ed77e27737a708d0d293fbf5",
  30. },
  31. true,
  32. "a5a665ff33ec",
  33. imageHeader,
  34. ctx.Image,
  35. },
  36. {types.Container{
  37. Image: "a5a665ff33eced1e0803148700880edab4",
  38. ImageID: "a5a665ff33eced1e0803148700880edab4269067ed77e27737a708d0d293fbf5",
  39. },
  40. false,
  41. "a5a665ff33eced1e0803148700880edab4",
  42. imageHeader,
  43. ctx.Image,
  44. },
  45. {types.Container{Image: ""}, true, "<no image>", imageHeader, ctx.Image},
  46. {types.Container{Command: "sh -c 'ls -la'"}, true, `"sh -c 'ls -la'"`, commandHeader, ctx.Command},
  47. {types.Container{Created: unix}, true, time.Unix(unix, 0).String(), createdAtHeader, ctx.CreatedAt},
  48. {types.Container{Ports: []types.Port{{PrivatePort: 8080, PublicPort: 8080, Type: "tcp"}}}, true, "8080/tcp", portsHeader, ctx.Ports},
  49. {types.Container{Status: "RUNNING"}, true, "RUNNING", statusHeader, ctx.Status},
  50. {types.Container{SizeRw: 10}, true, "10 B", sizeHeader, ctx.Size},
  51. {types.Container{SizeRw: 10, SizeRootFs: 20}, true, "10 B (virtual 20 B)", sizeHeader, ctx.Size},
  52. {types.Container{}, true, "", labelsHeader, ctx.Labels},
  53. {types.Container{Labels: map[string]string{"cpu": "6", "storage": "ssd"}}, true, "cpu=6,storage=ssd", labelsHeader, ctx.Labels},
  54. {types.Container{Created: unix}, true, "About a minute", runningForHeader, ctx.RunningFor},
  55. }
  56. for _, c := range cases {
  57. ctx = containerContext{c: c.container, trunc: c.trunc}
  58. v := c.call()
  59. if strings.Contains(v, ",") {
  60. compareMultipleValues(t, v, c.expValue)
  61. } else if v != c.expValue {
  62. t.Fatalf("Expected %s, was %s\n", c.expValue, v)
  63. }
  64. h := ctx.fullHeader()
  65. if h != c.expHeader {
  66. t.Fatalf("Expected %s, was %s\n", c.expHeader, h)
  67. }
  68. }
  69. c1 := types.Container{Labels: map[string]string{"com.docker.swarm.swarm-id": "33", "com.docker.swarm.node_name": "ubuntu"}}
  70. ctx = containerContext{c: c1, trunc: true}
  71. sid := ctx.Label("com.docker.swarm.swarm-id")
  72. node := ctx.Label("com.docker.swarm.node_name")
  73. if sid != "33" {
  74. t.Fatalf("Expected 33, was %s\n", sid)
  75. }
  76. if node != "ubuntu" {
  77. t.Fatalf("Expected ubuntu, was %s\n", node)
  78. }
  79. h := ctx.fullHeader()
  80. if h != "SWARM ID\tNODE NAME" {
  81. t.Fatalf("Expected %s, was %s\n", "SWARM ID\tNODE NAME", h)
  82. }
  83. c2 := types.Container{}
  84. ctx = containerContext{c: c2, trunc: true}
  85. label := ctx.Label("anything.really")
  86. if label != "" {
  87. t.Fatalf("Expected an empty string, was %s", label)
  88. }
  89. ctx = containerContext{c: c2, trunc: true}
  90. fullHeader := ctx.fullHeader()
  91. if fullHeader != "" {
  92. t.Fatalf("Expected fullHeader to be empty, was %s", fullHeader)
  93. }
  94. }
  95. func TestImagesContext(t *testing.T) {
  96. imageID := stringid.GenerateRandomID()
  97. unix := time.Now().Unix()
  98. var ctx imageContext
  99. cases := []struct {
  100. imageCtx imageContext
  101. expValue string
  102. expHeader string
  103. call func() string
  104. }{
  105. {imageContext{
  106. i: types.Image{ID: imageID},
  107. trunc: true,
  108. }, stringid.TruncateID(imageID), imageIDHeader, ctx.ID},
  109. {imageContext{
  110. i: types.Image{ID: imageID},
  111. trunc: false,
  112. }, imageID, imageIDHeader, ctx.ID},
  113. {imageContext{
  114. i: types.Image{Size: 10},
  115. trunc: true,
  116. }, "10 B", sizeHeader, ctx.Size},
  117. {imageContext{
  118. i: types.Image{Created: unix},
  119. trunc: true,
  120. }, time.Unix(unix, 0).String(), createdAtHeader, ctx.CreatedAt},
  121. // FIXME
  122. // {imageContext{
  123. // i: types.Image{Created: unix},
  124. // trunc: true,
  125. // }, units.HumanDuration(time.Unix(unix, 0)), createdSinceHeader, ctx.CreatedSince},
  126. {imageContext{
  127. i: types.Image{},
  128. repo: "busybox",
  129. }, "busybox", repositoryHeader, ctx.Repository},
  130. {imageContext{
  131. i: types.Image{},
  132. tag: "latest",
  133. }, "latest", tagHeader, ctx.Tag},
  134. {imageContext{
  135. i: types.Image{},
  136. digest: "sha256:d149ab53f8718e987c3a3024bb8aa0e2caadf6c0328f1d9d850b2a2a67f2819a",
  137. }, "sha256:d149ab53f8718e987c3a3024bb8aa0e2caadf6c0328f1d9d850b2a2a67f2819a", digestHeader, ctx.Digest},
  138. }
  139. for _, c := range cases {
  140. ctx = c.imageCtx
  141. v := c.call()
  142. if strings.Contains(v, ",") {
  143. compareMultipleValues(t, v, c.expValue)
  144. } else if v != c.expValue {
  145. t.Fatalf("Expected %s, was %s\n", c.expValue, v)
  146. }
  147. h := ctx.fullHeader()
  148. if h != c.expHeader {
  149. t.Fatalf("Expected %s, was %s\n", c.expHeader, h)
  150. }
  151. }
  152. }
  153. func compareMultipleValues(t *testing.T, value, expected string) {
  154. // comma-separated values means probably a map input, which won't
  155. // be guaranteed to have the same order as our expected value
  156. // We'll create maps and use reflect.DeepEquals to check instead:
  157. entriesMap := make(map[string]string)
  158. expMap := make(map[string]string)
  159. entries := strings.Split(value, ",")
  160. expectedEntries := strings.Split(expected, ",")
  161. for _, entry := range entries {
  162. keyval := strings.Split(entry, "=")
  163. entriesMap[keyval[0]] = keyval[1]
  164. }
  165. for _, expected := range expectedEntries {
  166. keyval := strings.Split(expected, "=")
  167. expMap[keyval[0]] = keyval[1]
  168. }
  169. if !reflect.DeepEqual(expMap, entriesMap) {
  170. t.Fatalf("Expected entries: %v, got: %v", expected, value)
  171. }
  172. }