diff --git a/cmd/compose/ps.go b/cmd/compose/ps.go index 5e89833d7..28ca0a747 100644 --- a/cmd/compose/ps.go +++ b/cmd/compose/ps.go @@ -24,6 +24,7 @@ import ( "sort" "strings" + "github.com/pkg/errors" "github.com/spf13/cobra" "github.com/docker/compose-cli/cli/formatter" @@ -37,24 +38,52 @@ type psOptions struct { All bool Quiet bool Services bool + Filter string + Status string +} + +func (p *psOptions) parseFilter() error { + if p.Filter == "" { + return nil + } + parts := strings.SplitN(p.Filter, "=", 2) + if len(parts) != 2 { + return errors.New("arguments to --filter should be in form KEY=VAL") + } + switch parts[0] { + case "status": + p.Status = parts[1] + case "source": + return api.ErrNotImplemented + default: + return fmt.Errorf("unknow filter %s", parts[0]) + } + return nil } func psCommand(p *projectOptions, backend api.Service) *cobra.Command { opts := psOptions{ projectOptions: p, } - psCmd := &cobra.Command{ + cmd := &cobra.Command{ Use: "ps", Short: "List containers", + PreRunE: func(cmd *cobra.Command, args []string) error { + return opts.parseFilter() + }, RunE: Adapt(func(ctx context.Context, args []string) error { return runPs(ctx, backend, args, opts) }), } - psCmd.Flags().StringVar(&opts.Format, "format", "pretty", "Format the output. Values: [pretty | json].") - psCmd.Flags().BoolVarP(&opts.Quiet, "quiet", "q", false, "Only display IDs") - psCmd.Flags().BoolVar(&opts.Services, "services", false, "Display services") - psCmd.Flags().BoolVarP(&opts.All, "all", "a", false, "Show all stopped containers (including those created by the run command)") - return psCmd + flags := cmd.Flags() + flags.StringVar(&opts.Format, "format", "pretty", "Format the output. Values: [pretty | json]") + flags.StringVar(&opts.Filter, "filter", "", "Filter services by a property") + flags.StringVar(&opts.Status, "status", "", "Filter services by status") + flags.BoolVarP(&opts.Quiet, "quiet", "q", false, "Only display IDs") + flags.BoolVar(&opts.Services, "services", false, "Display services") + flags.BoolVarP(&opts.All, "all", "a", false, "Show all stopped containers (including those created by the run command)") + flags.Lookup("filter").Hidden = true + return cmd } func runPs(ctx context.Context, backend api.Service, services []string, opts psOptions) error { @@ -87,6 +116,10 @@ func runPs(ctx context.Context, backend api.Service, services []string, opts psO return nil } + if opts.Status != "" { + containers = filterByStatus(containers, opts.Status) + } + sort.Slice(containers, func(i, j int) bool { return containers[i].Name < containers[j].Name }) @@ -113,3 +146,19 @@ func runPs(ctx context.Context, backend api.Service, services []string, opts psO }, "NAME", "SERVICE", "STATUS", "PORTS") } + +func filterByStatus(containers []api.ContainerSummary, status string) []api.ContainerSummary { + hasContainerWithState := map[string]struct{}{} + for _, c := range containers { + if c.State == status { + hasContainerWithState[c.Service] = struct{}{} + } + } + var filtered []api.ContainerSummary + for _, c := range containers { + if _, ok := hasContainerWithState[c.Service]; ok { + filtered = append(filtered, c) + } + } + return filtered +}