194 lines
5.0 KiB
Go
194 lines
5.0 KiB
Go
package command
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/hashicorp/consul/api"
|
|
"github.com/hashicorp/consul/command/flags"
|
|
"github.com/mitchellh/cli"
|
|
"github.com/ryanuber/columnize"
|
|
)
|
|
|
|
var _ cli.Command = (*CatalogListNodesCommand)(nil)
|
|
|
|
// CatalogListNodesCommand is a Command implementation that is used to fetch all the
|
|
// nodes in the catalog.
|
|
type CatalogListNodesCommand struct {
|
|
BaseCommand
|
|
|
|
// flags
|
|
detailed bool
|
|
near string
|
|
nodeMeta map[string]string
|
|
service string
|
|
}
|
|
|
|
func (c *CatalogListNodesCommand) initFlags() {
|
|
c.InitFlagSet()
|
|
c.FlagSet.BoolVar(&c.detailed, "detailed", false, "Output detailed information about "+
|
|
"the nodes including their addresses and metadata.")
|
|
c.FlagSet.StringVar(&c.near, "near", "", "Node name to sort the node list in ascending "+
|
|
"order based on estimated round-trip time from that node. "+
|
|
"Passing \"_agent\" will use this agent's node for sorting.")
|
|
c.FlagSet.Var((*flags.FlagMapValue)(&c.nodeMeta), "node-meta", "Metadata to "+
|
|
"filter nodes with the given `key=value` pairs. This flag may be "+
|
|
"specified multiple times to filter on multiple sources of metadata.")
|
|
c.FlagSet.StringVar(&c.service, "service", "", "Service `id or name` to filter nodes. "+
|
|
"Only nodes which are providing the given service will be returned.")
|
|
}
|
|
|
|
func (c *CatalogListNodesCommand) Help() string {
|
|
c.initFlags()
|
|
return c.HelpCommand(`
|
|
Usage: consul catalog nodes [options]
|
|
|
|
Retrieves the list nodes registered in a given datacenter. By default, the
|
|
datacenter of the local agent is queried.
|
|
|
|
To retrieve the list of nodes:
|
|
|
|
$ consul catalog nodes
|
|
|
|
To print detailed information including full node IDs, tagged addresses, and
|
|
metadata information:
|
|
|
|
$ consul catalog nodes -detailed
|
|
|
|
To list nodes which are running a particular service:
|
|
|
|
$ consul catalog nodes -service=web
|
|
|
|
To filter by node metadata:
|
|
|
|
$ consul catalog nodes -node-meta="foo=bar"
|
|
|
|
To sort nodes by estimated round-trip time from node-web:
|
|
|
|
$ consul catalog nodes -near=node-web
|
|
|
|
For a full list of options and examples, please see the Consul documentation.
|
|
|
|
`)
|
|
}
|
|
|
|
func (c *CatalogListNodesCommand) Run(args []string) int {
|
|
c.initFlags()
|
|
if err := c.FlagSet.Parse(args); err != nil {
|
|
return 1
|
|
}
|
|
|
|
if l := len(c.FlagSet.Args()); l > 0 {
|
|
c.UI.Error(fmt.Sprintf("Too many arguments (expected 0, got %d)", l))
|
|
return 1
|
|
}
|
|
|
|
// Create and test the HTTP client
|
|
client, err := c.HTTPClient()
|
|
if err != nil {
|
|
c.UI.Error(fmt.Sprintf("Error connecting to Consul agent: %s", err))
|
|
return 1
|
|
}
|
|
|
|
var nodes []*api.Node
|
|
if c.service != "" {
|
|
services, _, err := client.Catalog().Service(c.service, "", &api.QueryOptions{
|
|
Near: c.near,
|
|
NodeMeta: c.nodeMeta,
|
|
})
|
|
if err != nil {
|
|
c.UI.Error(fmt.Sprintf("Error listing nodes for service: %s", err))
|
|
return 1
|
|
}
|
|
|
|
nodes = make([]*api.Node, len(services))
|
|
for i, s := range services {
|
|
nodes[i] = &api.Node{
|
|
ID: s.ID,
|
|
Node: s.Node,
|
|
Address: s.Address,
|
|
Datacenter: s.Datacenter,
|
|
TaggedAddresses: s.TaggedAddresses,
|
|
Meta: s.NodeMeta,
|
|
CreateIndex: s.CreateIndex,
|
|
ModifyIndex: s.ModifyIndex,
|
|
}
|
|
}
|
|
} else {
|
|
nodes, _, err = client.Catalog().Nodes(&api.QueryOptions{
|
|
Near: c.near,
|
|
NodeMeta: c.nodeMeta,
|
|
})
|
|
if err != nil {
|
|
c.UI.Error(fmt.Sprintf("Error listing nodes: %s", err))
|
|
return 1
|
|
}
|
|
}
|
|
|
|
// Handle the edge case where there are no nodes that match the query.
|
|
if len(nodes) == 0 {
|
|
c.UI.Error("No nodes match the given query - try expanding your search.")
|
|
return 0
|
|
}
|
|
|
|
output, err := printNodes(nodes, c.detailed)
|
|
if err != nil {
|
|
c.UI.Error(fmt.Sprintf("Error printing nodes: %s", err))
|
|
return 1
|
|
}
|
|
|
|
c.UI.Info(output)
|
|
|
|
return 0
|
|
}
|
|
|
|
func (c *CatalogListNodesCommand) Synopsis() string {
|
|
return "Lists all nodes in the given datacenter"
|
|
}
|
|
|
|
// printNodes accepts a list of nodes and prints information in a tabular
|
|
// format about the nodes.
|
|
func printNodes(nodes []*api.Node, detailed bool) (string, error) {
|
|
var result []string
|
|
if detailed {
|
|
result = detailedNodes(nodes)
|
|
} else {
|
|
result = simpleNodes(nodes)
|
|
}
|
|
|
|
return columnize.SimpleFormat(result), nil
|
|
}
|
|
|
|
func detailedNodes(nodes []*api.Node) []string {
|
|
result := make([]string, 0, len(nodes)+1)
|
|
header := "Node|ID|Address|DC|TaggedAddresses|Meta"
|
|
result = append(result, header)
|
|
|
|
for _, node := range nodes {
|
|
result = append(result, fmt.Sprintf("%s|%s|%s|%s|%s|%s",
|
|
node.Node, node.ID, node.Address, node.Datacenter,
|
|
mapToKV(node.TaggedAddresses, ", "), mapToKV(node.Meta, ", ")))
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func simpleNodes(nodes []*api.Node) []string {
|
|
result := make([]string, 0, len(nodes)+1)
|
|
header := "Node|ID|Address|DC"
|
|
result = append(result, header)
|
|
|
|
for _, node := range nodes {
|
|
// Shorten the ID in non-detailed mode to just the first octet.
|
|
id := node.ID
|
|
idx := strings.Index(id, "-")
|
|
if idx > 0 {
|
|
id = id[0:idx]
|
|
}
|
|
result = append(result, fmt.Sprintf("%s|%s|%s|%s",
|
|
node.Node, id, node.Address, node.Datacenter))
|
|
}
|
|
|
|
return result
|
|
}
|