2017-10-11 14:46:12 +00:00
|
|
|
package catlistsvc
|
2017-07-14 19:45:08 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2017-10-11 14:46:12 +00:00
|
|
|
"flag"
|
2017-07-14 19:45:08 +00:00
|
|
|
"fmt"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
"text/tabwriter"
|
|
|
|
|
|
|
|
"github.com/hashicorp/consul/api"
|
2017-10-11 12:51:19 +00:00
|
|
|
"github.com/hashicorp/consul/command/flags"
|
2017-07-14 19:45:08 +00:00
|
|
|
"github.com/mitchellh/cli"
|
|
|
|
)
|
|
|
|
|
2017-10-11 14:46:12 +00:00
|
|
|
func New(ui cli.Ui) *cmd {
|
|
|
|
c := &cmd{UI: ui}
|
2017-10-11 21:37:15 +00:00
|
|
|
c.init()
|
2017-10-11 14:46:12 +00:00
|
|
|
return c
|
|
|
|
}
|
2017-07-14 19:45:08 +00:00
|
|
|
|
2017-10-11 14:46:12 +00:00
|
|
|
type cmd struct {
|
|
|
|
UI cli.Ui
|
|
|
|
flags *flag.FlagSet
|
|
|
|
http *flags.HTTPFlags
|
2017-10-11 21:15:53 +00:00
|
|
|
usage string
|
2017-10-11 12:51:18 +00:00
|
|
|
|
|
|
|
// flags
|
|
|
|
node string
|
|
|
|
nodeMeta map[string]string
|
|
|
|
tags bool
|
|
|
|
}
|
|
|
|
|
2017-10-11 21:37:15 +00:00
|
|
|
func (c *cmd) init() {
|
2017-10-11 14:46:12 +00:00
|
|
|
c.flags = flag.NewFlagSet("", flag.ContinueOnError)
|
|
|
|
c.flags.StringVar(&c.node, "node", "",
|
2017-10-11 12:51:18 +00:00
|
|
|
"Node `id or name` for which to list services.")
|
2017-10-11 14:46:12 +00:00
|
|
|
c.flags.Var((*flags.FlagMapValue)(&c.nodeMeta), "node-meta", "Metadata to "+
|
2017-10-11 12:51:18 +00:00
|
|
|
"filter nodes with the given `key=value` pairs. If specified, only "+
|
|
|
|
"services running on nodes matching the given metadata will be returned. "+
|
|
|
|
"This flag may be specified multiple times to filter on multiple sources "+
|
|
|
|
"of metadata.")
|
2017-10-11 14:46:12 +00:00
|
|
|
c.flags.BoolVar(&c.tags, "tags", false, "Display each service's tags as a "+
|
2017-10-11 12:51:18 +00:00
|
|
|
"comma-separated list beside each service entry.")
|
2017-07-14 19:45:08 +00:00
|
|
|
|
2017-10-11 14:46:12 +00:00
|
|
|
c.http = &flags.HTTPFlags{}
|
|
|
|
flags.Merge(c.flags, c.http.ClientFlags())
|
|
|
|
flags.Merge(c.flags, c.http.ServerFlags())
|
2017-10-11 21:15:53 +00:00
|
|
|
c.usage = flags.Usage(usage, c.flags, c.http.ClientFlags(), c.http.ServerFlags())
|
2017-07-14 19:45:08 +00:00
|
|
|
}
|
|
|
|
|
2017-10-11 14:46:12 +00:00
|
|
|
func (c *cmd) Run(args []string) int {
|
|
|
|
if err := c.flags.Parse(args); err != nil {
|
2017-07-14 19:45:08 +00:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2017-10-11 14:46:12 +00:00
|
|
|
if l := len(c.flags.Args()); l > 0 {
|
2017-07-14 19:45:08 +00:00
|
|
|
c.UI.Error(fmt.Sprintf("Too many arguments (expected 0, got %d)", l))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create and test the HTTP client
|
2017-10-11 14:46:12 +00:00
|
|
|
client, err := c.http.APIClient()
|
2017-07-14 19:45:08 +00:00
|
|
|
if err != nil {
|
|
|
|
c.UI.Error(fmt.Sprintf("Error connecting to Consul agent: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
var services map[string][]string
|
2017-10-11 12:51:18 +00:00
|
|
|
if c.node != "" {
|
|
|
|
catalogNode, _, err := client.Catalog().Node(c.node, &api.QueryOptions{
|
|
|
|
NodeMeta: c.nodeMeta,
|
2017-07-14 19:45:08 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
c.UI.Error(fmt.Sprintf("Error listing services for node: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
if catalogNode != nil {
|
|
|
|
services = make(map[string][]string, len(catalogNode.Services))
|
|
|
|
for _, s := range catalogNode.Services {
|
|
|
|
services[s.Service] = append(services[s.Service], s.Tags...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
services, _, err = client.Catalog().Services(&api.QueryOptions{
|
2017-10-11 12:51:18 +00:00
|
|
|
NodeMeta: c.nodeMeta,
|
2017-07-14 19:45:08 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
c.UI.Error(fmt.Sprintf("Error listing services: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle the edge case where there are no services that match the query.
|
|
|
|
if len(services) == 0 {
|
|
|
|
c.UI.Error("No services match the given query - try expanding your search.")
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
// Order the map for consistent output
|
|
|
|
order := make([]string, 0, len(services))
|
|
|
|
for k, _ := range services {
|
|
|
|
order = append(order, k)
|
|
|
|
}
|
|
|
|
sort.Strings(order)
|
|
|
|
|
2017-10-11 12:51:18 +00:00
|
|
|
if c.tags {
|
2017-07-14 19:45:08 +00:00
|
|
|
var b bytes.Buffer
|
|
|
|
tw := tabwriter.NewWriter(&b, 0, 2, 6, ' ', 0)
|
|
|
|
for _, s := range order {
|
2017-07-15 00:00:08 +00:00
|
|
|
sort.Strings(services[s])
|
2017-07-14 19:45:08 +00:00
|
|
|
fmt.Fprintf(tw, "%s\t%s\n", s, strings.Join(services[s], ","))
|
|
|
|
}
|
|
|
|
if err := tw.Flush(); err != nil {
|
|
|
|
c.UI.Error(fmt.Sprintf("Error flushing tabwriter: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
c.UI.Output(strings.TrimSpace(b.String()))
|
|
|
|
} else {
|
|
|
|
for _, s := range order {
|
|
|
|
c.UI.Output(s)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2017-10-11 14:46:12 +00:00
|
|
|
func (c *cmd) Synopsis() string {
|
2017-07-14 19:45:08 +00:00
|
|
|
return "Lists all registered services in a datacenter"
|
|
|
|
}
|
2017-10-11 14:46:12 +00:00
|
|
|
|
|
|
|
func (c *cmd) Help() string {
|
2017-10-11 21:15:53 +00:00
|
|
|
return c.usage
|
|
|
|
}
|
|
|
|
|
|
|
|
const usage = `Usage: consul catalog services [options]
|
2017-10-11 14:46:12 +00:00
|
|
|
|
|
|
|
Retrieves the list services registered in a given datacenter. By default, the
|
|
|
|
datacenter of the local agent is queried.
|
|
|
|
|
|
|
|
To retrieve the list of services:
|
|
|
|
|
|
|
|
$ consul catalog services
|
|
|
|
|
|
|
|
To include the services' tags in the output:
|
|
|
|
|
|
|
|
$ consul catalog services -tags
|
|
|
|
|
|
|
|
To list services which run on a particular node:
|
|
|
|
|
|
|
|
$ consul catalog services -node=web
|
|
|
|
|
|
|
|
To filter services on node metadata:
|
|
|
|
|
|
|
|
$ consul catalog services -node-meta="foo=bar"
|
|
|
|
|
|
|
|
For a full list of options and examples, please see the Consul documentation.`
|