| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249 |
- /*
- Copyright 2020 Docker Compose CLI authors
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- */
- package compose
- import (
- "context"
- "fmt"
- "io"
- "os"
- "sort"
- "strconv"
- "strings"
- "github.com/docker/compose/v2/cmd/formatter"
- "github.com/docker/compose/v2/pkg/utils"
- formatter2 "github.com/docker/cli/cli/command/formatter"
- "github.com/pkg/errors"
- "github.com/spf13/cobra"
- "github.com/docker/compose/v2/pkg/api"
- )
- type psOptions struct {
- *projectOptions
- Format string
- 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 = append(p.Status, parts[1])
- case "source":
- return api.ErrNotImplemented
- default:
- return fmt.Errorf("unknown filter %s", parts[0])
- }
- return nil
- }
- func psCommand(p *projectOptions, backend api.Service) *cobra.Command {
- opts := psOptions{
- projectOptions: p,
- }
- psCmd := &cobra.Command{
- Use: "ps [SERVICE...]",
- 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)
- }),
- ValidArgsFunction: serviceCompletion(p),
- }
- flags := psCmd.Flags()
- flags.StringVar(&opts.Format, "format", "pretty", "Format the output. Values: [pretty | json]")
- flags.StringVar(&opts.Filter, "filter", "", "Filter services by a property (supported filters: status).")
- flags.StringArrayVar(&opts.Status, "status", []string{}, "Filter services by status. Values: [paused | restarting | removing | running | dead | created | exited]")
- 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)")
- return psCmd
- }
- func runPs(ctx context.Context, backend api.Service, services []string, opts psOptions) error {
- projectName, err := opts.toProjectName()
- if err != nil {
- return err
- }
- containers, err := backend.Ps(ctx, projectName, api.PsOptions{
- All: opts.All,
- Services: services,
- })
- if err != nil {
- return err
- }
- SERVICES:
- for _, s := range services {
- for _, c := range containers {
- if c.Service == s {
- continue SERVICES
- }
- }
- return fmt.Errorf("no such service: %s", s)
- }
- if len(opts.Status) != 0 {
- containers = filterByStatus(containers, opts.Status)
- }
- sort.Slice(containers, func(i, j int) bool {
- return containers[i].Name < containers[j].Name
- })
- if opts.Quiet {
- for _, c := range containers {
- fmt.Println(c.ID)
- }
- return nil
- }
- if opts.Services {
- services := []string{}
- for _, s := range containers {
- if !utils.StringContains(services, s.Service) {
- services = append(services, s.Service)
- }
- }
- fmt.Println(strings.Join(services, "\n"))
- return nil
- }
- return formatter.Print(containers, opts.Format, os.Stdout,
- writter(containers),
- "NAME", "COMMAND", "SERVICE", "STATUS", "PORTS")
- }
- func writter(containers []api.ContainerSummary) func(w io.Writer) {
- return func(w io.Writer) {
- for _, container := range containers {
- ports := DisplayablePorts(container)
- status := container.State
- if status == "running" && container.Health != "" {
- status = fmt.Sprintf("%s (%s)", container.State, container.Health)
- } else if status == "exited" || status == "dead" {
- status = fmt.Sprintf("%s (%d)", container.State, container.ExitCode)
- }
- command := formatter2.Ellipsis(container.Command, 20)
- _, _ = fmt.Fprintf(w, "%s\t%s\t%s\t%s\t%s\n", container.Name, strconv.Quote(command), container.Service, status, ports)
- }
- }
- }
- func filterByStatus(containers []api.ContainerSummary, statuses []string) []api.ContainerSummary {
- var filtered []api.ContainerSummary
- for _, c := range containers {
- if hasStatus(c, statuses) {
- filtered = append(filtered, c)
- }
- }
- return filtered
- }
- func hasStatus(c api.ContainerSummary, statuses []string) bool {
- for _, status := range statuses {
- if c.State == status {
- return true
- }
- }
- return false
- }
- type portRange struct {
- pStart int
- pEnd int
- tStart int
- tEnd int
- IP string
- protocol string
- }
- func (pr portRange) String() string {
- var (
- pub string
- tgt string
- )
- if pr.pEnd > pr.pStart {
- pub = fmt.Sprintf("%s:%d-%d->", pr.IP, pr.pStart, pr.pEnd)
- } else if pr.pStart > 0 {
- pub = fmt.Sprintf("%s:%d->", pr.IP, pr.pStart)
- }
- if pr.tEnd > pr.tStart {
- tgt = fmt.Sprintf("%d-%d", pr.tStart, pr.tEnd)
- } else {
- tgt = fmt.Sprintf("%d", pr.tStart)
- }
- return fmt.Sprintf("%s%s/%s", pub, tgt, pr.protocol)
- }
- // DisplayablePorts is copy pasted from https://github.com/docker/cli/pull/581/files
- func DisplayablePorts(c api.ContainerSummary) string {
- if c.Publishers == nil {
- return ""
- }
- sort.Sort(c.Publishers)
- pr := portRange{}
- ports := []string{}
- for _, p := range c.Publishers {
- prIsRange := pr.tEnd != pr.tStart
- tOverlaps := p.TargetPort <= pr.tEnd
- // Start a new port-range if:
- // - the protocol is different from the current port-range
- // - published or target port are not consecutive to the current port-range
- // - the current port-range is a _range_, and the target port overlaps with the current range's target-ports
- if p.Protocol != pr.protocol || p.URL != pr.IP || p.PublishedPort-pr.pEnd > 1 || p.TargetPort-pr.tEnd > 1 || prIsRange && tOverlaps {
- // start a new port-range, and print the previous port-range (if any)
- if pr.pStart > 0 {
- ports = append(ports, pr.String())
- }
- pr = portRange{
- pStart: p.PublishedPort,
- pEnd: p.PublishedPort,
- tStart: p.TargetPort,
- tEnd: p.TargetPort,
- protocol: p.Protocol,
- IP: p.URL,
- }
- continue
- }
- pr.pEnd = p.PublishedPort
- pr.tEnd = p.TargetPort
- }
- if pr.tStart > 0 {
- ports = append(ports, pr.String())
- }
- return strings.Join(ports, ", ")
- }
|