custom.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. package ps
  2. import (
  3. "bytes"
  4. "fmt"
  5. "strconv"
  6. "strings"
  7. "text/tabwriter"
  8. "text/template"
  9. "time"
  10. "github.com/docker/docker/api"
  11. "github.com/docker/docker/api/types"
  12. "github.com/docker/docker/pkg/stringid"
  13. "github.com/docker/docker/pkg/stringutils"
  14. "github.com/docker/docker/pkg/units"
  15. )
  16. const (
  17. tableKey = "table"
  18. idHeader = "CONTAINER ID"
  19. imageHeader = "IMAGE"
  20. namesHeader = "NAMES"
  21. commandHeader = "COMMAND"
  22. createdAtHeader = "CREATED AT"
  23. runningForHeader = "CREATED"
  24. statusHeader = "STATUS"
  25. portsHeader = "PORTS"
  26. sizeHeader = "SIZE"
  27. labelsHeader = "LABELS"
  28. )
  29. type containerContext struct {
  30. trunc bool
  31. header []string
  32. c types.Container
  33. }
  34. func (c *containerContext) ID() string {
  35. c.addHeader(idHeader)
  36. if c.trunc {
  37. return stringid.TruncateID(c.c.ID)
  38. }
  39. return c.c.ID
  40. }
  41. func (c *containerContext) Names() string {
  42. c.addHeader(namesHeader)
  43. names := stripNamePrefix(c.c.Names)
  44. if c.trunc {
  45. for _, name := range names {
  46. if len(strings.Split(name, "/")) == 1 {
  47. names = []string{name}
  48. break
  49. }
  50. }
  51. }
  52. return strings.Join(names, ",")
  53. }
  54. func (c *containerContext) Image() string {
  55. c.addHeader(imageHeader)
  56. if c.c.Image == "" {
  57. return "<no image>"
  58. }
  59. return c.c.Image
  60. }
  61. func (c *containerContext) Command() string {
  62. c.addHeader(commandHeader)
  63. command := c.c.Command
  64. if c.trunc {
  65. command = stringutils.Truncate(command, 20)
  66. }
  67. return strconv.Quote(command)
  68. }
  69. func (c *containerContext) CreatedAt() string {
  70. c.addHeader(createdAtHeader)
  71. return time.Unix(int64(c.c.Created), 0).String()
  72. }
  73. func (c *containerContext) RunningFor() string {
  74. c.addHeader(runningForHeader)
  75. createdAt := time.Unix(int64(c.c.Created), 0)
  76. return units.HumanDuration(time.Now().UTC().Sub(createdAt))
  77. }
  78. func (c *containerContext) Ports() string {
  79. c.addHeader(portsHeader)
  80. return api.DisplayablePorts(c.c.Ports)
  81. }
  82. func (c *containerContext) Status() string {
  83. c.addHeader(statusHeader)
  84. return c.c.Status
  85. }
  86. func (c *containerContext) Size() string {
  87. c.addHeader(sizeHeader)
  88. srw := units.HumanSize(float64(c.c.SizeRw))
  89. sv := units.HumanSize(float64(c.c.SizeRootFs))
  90. sf := srw
  91. if c.c.SizeRootFs > 0 {
  92. sf = fmt.Sprintf("%s (virtual %s)", srw, sv)
  93. }
  94. return sf
  95. }
  96. func (c *containerContext) Labels() string {
  97. c.addHeader(labelsHeader)
  98. if c.c.Labels == nil {
  99. return ""
  100. }
  101. var joinLabels []string
  102. for k, v := range c.c.Labels {
  103. joinLabels = append(joinLabels, fmt.Sprintf("%s=%s", k, v))
  104. }
  105. return strings.Join(joinLabels, ",")
  106. }
  107. func (c *containerContext) Label(name string) string {
  108. n := strings.Split(name, ".")
  109. r := strings.NewReplacer("-", " ", "_", " ")
  110. h := r.Replace(n[len(n)-1])
  111. c.addHeader(h)
  112. if c.c.Labels == nil {
  113. return ""
  114. }
  115. return c.c.Labels[name]
  116. }
  117. func (c *containerContext) fullHeader() string {
  118. if c.header == nil {
  119. return ""
  120. }
  121. return strings.Join(c.header, "\t")
  122. }
  123. func (c *containerContext) addHeader(header string) {
  124. if c.header == nil {
  125. c.header = []string{}
  126. }
  127. c.header = append(c.header, strings.ToUpper(header))
  128. }
  129. func customFormat(ctx Context, containers []types.Container) {
  130. var (
  131. table bool
  132. header string
  133. format = ctx.Format
  134. buffer = bytes.NewBufferString("")
  135. )
  136. if strings.HasPrefix(ctx.Format, tableKey) {
  137. table = true
  138. format = format[len(tableKey):]
  139. }
  140. format = strings.Trim(format, " ")
  141. r := strings.NewReplacer(`\t`, "\t", `\n`, "\n")
  142. format = r.Replace(format)
  143. if table && ctx.Size {
  144. format += "\t{{.Size}}"
  145. }
  146. tmpl, err := template.New("ps template").Parse(format)
  147. if err != nil {
  148. buffer.WriteString(fmt.Sprintf("Invalid `docker ps` format: %v\n", err))
  149. }
  150. for _, container := range containers {
  151. containerCtx := &containerContext{
  152. trunc: ctx.Trunc,
  153. c: container,
  154. }
  155. if err := tmpl.Execute(buffer, containerCtx); err != nil {
  156. buffer = bytes.NewBufferString(fmt.Sprintf("Invalid `docker ps` format: %v\n", err))
  157. break
  158. }
  159. if table && len(header) == 0 {
  160. header = containerCtx.fullHeader()
  161. }
  162. buffer.WriteString("\n")
  163. }
  164. if table {
  165. if len(header) == 0 {
  166. // if we still don't have a header, we didn't have any containers so we need to fake it to get the right headers from the template
  167. containerCtx := &containerContext{}
  168. tmpl.Execute(bytes.NewBufferString(""), containerCtx)
  169. header = containerCtx.fullHeader()
  170. }
  171. t := tabwriter.NewWriter(ctx.Output, 20, 1, 3, ' ', 0)
  172. t.Write([]byte(header))
  173. t.Write([]byte("\n"))
  174. buffer.WriteTo(t)
  175. t.Flush()
  176. } else {
  177. buffer.WriteTo(ctx.Output)
  178. }
  179. }
  180. func stripNamePrefix(ss []string) []string {
  181. for i, s := range ss {
  182. ss[i] = s[1:]
  183. }
  184. return ss
  185. }