ps.go 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. package service
  2. import (
  3. "fmt"
  4. "strings"
  5. "golang.org/x/net/context"
  6. "github.com/docker/docker/api/types"
  7. "github.com/docker/docker/api/types/filters"
  8. "github.com/docker/docker/cli"
  9. "github.com/docker/docker/cli/command"
  10. "github.com/docker/docker/cli/command/idresolver"
  11. "github.com/docker/docker/cli/command/node"
  12. "github.com/docker/docker/cli/command/task"
  13. "github.com/docker/docker/opts"
  14. "github.com/spf13/cobra"
  15. )
  16. type psOptions struct {
  17. services []string
  18. quiet bool
  19. noResolve bool
  20. noTrunc bool
  21. filter opts.FilterOpt
  22. }
  23. func newPsCommand(dockerCli *command.DockerCli) *cobra.Command {
  24. opts := psOptions{filter: opts.NewFilterOpt()}
  25. cmd := &cobra.Command{
  26. Use: "ps [OPTIONS] SERVICE [SERVICE...]",
  27. Short: "List the tasks of one or more services",
  28. Args: cli.RequiresMinArgs(1),
  29. RunE: func(cmd *cobra.Command, args []string) error {
  30. opts.services = args
  31. return runPS(dockerCli, opts)
  32. },
  33. }
  34. flags := cmd.Flags()
  35. flags.BoolVarP(&opts.quiet, "quiet", "q", false, "Only display task IDs")
  36. flags.BoolVar(&opts.noTrunc, "no-trunc", false, "Do not truncate output")
  37. flags.BoolVar(&opts.noResolve, "no-resolve", false, "Do not map IDs to Names")
  38. flags.VarP(&opts.filter, "filter", "f", "Filter output based on conditions provided")
  39. return cmd
  40. }
  41. func runPS(dockerCli *command.DockerCli, opts psOptions) error {
  42. client := dockerCli.Client()
  43. ctx := context.Background()
  44. filter := opts.filter.Value()
  45. serviceIDFilter := filters.NewArgs()
  46. serviceNameFilter := filters.NewArgs()
  47. for _, service := range opts.services {
  48. serviceIDFilter.Add("id", service)
  49. serviceNameFilter.Add("name", service)
  50. }
  51. serviceByIDList, err := client.ServiceList(ctx, types.ServiceListOptions{Filters: serviceIDFilter})
  52. if err != nil {
  53. return err
  54. }
  55. serviceByNameList, err := client.ServiceList(ctx, types.ServiceListOptions{Filters: serviceNameFilter})
  56. if err != nil {
  57. return err
  58. }
  59. for _, service := range opts.services {
  60. serviceCount := 0
  61. // Lookup by ID/Prefix
  62. for _, serviceEntry := range serviceByIDList {
  63. if strings.HasPrefix(serviceEntry.ID, service) {
  64. filter.Add("service", serviceEntry.ID)
  65. serviceCount++
  66. }
  67. }
  68. // Lookup by Name/Prefix
  69. for _, serviceEntry := range serviceByNameList {
  70. if strings.HasPrefix(serviceEntry.Spec.Annotations.Name, service) {
  71. filter.Add("service", serviceEntry.ID)
  72. serviceCount++
  73. }
  74. }
  75. // If nothing has been found, return immediately.
  76. if serviceCount == 0 {
  77. return fmt.Errorf("no such services: %s", service)
  78. }
  79. }
  80. if filter.Include("node") {
  81. nodeFilters := filter.Get("node")
  82. for _, nodeFilter := range nodeFilters {
  83. nodeReference, err := node.Reference(ctx, client, nodeFilter)
  84. if err != nil {
  85. return err
  86. }
  87. filter.Del("node", nodeFilter)
  88. filter.Add("node", nodeReference)
  89. }
  90. }
  91. tasks, err := client.TaskList(ctx, types.TaskListOptions{Filters: filter})
  92. if err != nil {
  93. return err
  94. }
  95. if opts.quiet {
  96. return task.PrintQuiet(dockerCli, tasks)
  97. }
  98. return task.Print(dockerCli, ctx, tasks, idresolver.New(client, opts.noResolve), opts.noTrunc)
  99. }