| package service |
| |
| import ( |
| "fmt" |
| "io" |
| "strings" |
| "text/tabwriter" |
| |
| "github.com/docker/docker/api/client" |
| "github.com/docker/docker/cli" |
| "github.com/docker/docker/opts" |
| "github.com/docker/docker/pkg/stringid" |
| "github.com/docker/engine-api/types" |
| "github.com/docker/engine-api/types/filters" |
| "github.com/docker/engine-api/types/swarm" |
| "github.com/spf13/cobra" |
| "golang.org/x/net/context" |
| ) |
| |
| const ( |
| listItemFmt = "%s\t%s\t%s\t%s\t%s\n" |
| ) |
| |
| type listOptions struct { |
| quiet bool |
| filter opts.FilterOpt |
| } |
| |
| func newListCommand(dockerCli *client.DockerCli) *cobra.Command { |
| opts := listOptions{filter: opts.NewFilterOpt()} |
| |
| cmd := &cobra.Command{ |
| Use: "ls [OPTIONS]", |
| Aliases: []string{"list"}, |
| Short: "List services", |
| Args: cli.NoArgs, |
| RunE: func(cmd *cobra.Command, args []string) error { |
| return runList(dockerCli, opts) |
| }, |
| } |
| |
| flags := cmd.Flags() |
| flags.BoolVarP(&opts.quiet, "quiet", "q", false, "Only display IDs") |
| flags.VarP(&opts.filter, "filter", "f", "Filter output based on conditions provided") |
| |
| return cmd |
| } |
| |
| func runList(dockerCli *client.DockerCli, opts listOptions) error { |
| ctx := context.Background() |
| client := dockerCli.Client() |
| |
| services, err := client.ServiceList(ctx, types.ServiceListOptions{Filter: opts.filter.Value()}) |
| if err != nil { |
| return err |
| } |
| |
| out := dockerCli.Out() |
| if opts.quiet { |
| printQuiet(out, services) |
| } else { |
| taskFilter := filters.NewArgs() |
| for _, service := range services { |
| taskFilter.Add("service", service.ID) |
| } |
| |
| tasks, err := client.TaskList(ctx, types.TaskListOptions{Filter: taskFilter}) |
| if err != nil { |
| return err |
| } |
| |
| nodes, err := client.NodeList(ctx, types.NodeListOptions{}) |
| if err != nil { |
| return err |
| } |
| activeNodes := make(map[string]struct{}) |
| for _, n := range nodes { |
| if n.Status.State == swarm.NodeStateReady { |
| activeNodes[n.ID] = struct{}{} |
| } |
| } |
| |
| running := map[string]int{} |
| for _, task := range tasks { |
| if _, nodeActive := activeNodes[task.NodeID]; nodeActive && task.Status.State == "running" { |
| running[task.ServiceID]++ |
| } |
| } |
| |
| printTable(out, services, running) |
| } |
| return nil |
| } |
| |
| func printTable(out io.Writer, services []swarm.Service, running map[string]int) { |
| writer := tabwriter.NewWriter(out, 0, 4, 2, ' ', 0) |
| |
| // Ignore flushing errors |
| defer writer.Flush() |
| |
| fmt.Fprintf(writer, listItemFmt, "ID", "NAME", "REPLICAS", "IMAGE", "COMMAND") |
| for _, service := range services { |
| replicas := "" |
| if service.Spec.Mode.Replicated != nil && service.Spec.Mode.Replicated.Replicas != nil { |
| replicas = fmt.Sprintf("%d/%d", running[service.ID], *service.Spec.Mode.Replicated.Replicas) |
| } else if service.Spec.Mode.Global != nil { |
| replicas = "global" |
| } |
| fmt.Fprintf( |
| writer, |
| listItemFmt, |
| stringid.TruncateID(service.ID), |
| service.Spec.Name, |
| replicas, |
| service.Spec.TaskTemplate.ContainerSpec.Image, |
| strings.Join(service.Spec.TaskTemplate.ContainerSpec.Args, " ")) |
| } |
| } |
| |
| func printQuiet(out io.Writer, services []swarm.Service) { |
| for _, service := range services { |
| fmt.Fprintln(out, service.ID) |
| } |
| } |