2023-08-11 09:12:13 -04:00
|
|
|
// Copyright (c) HashiCorp, Inc.
|
|
|
|
// SPDX-License-Identifier: BUSL-1.1
|
|
|
|
|
2023-07-17 15:15:22 -07:00
|
|
|
package sprawl
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"sort"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"text/tabwriter"
|
|
|
|
)
|
|
|
|
|
|
|
|
// PrintDetails will dump relevant addressing and naming data to the logger for
|
|
|
|
// human interaction purposes.
|
|
|
|
func (s *Sprawl) PrintDetails() error {
|
|
|
|
det := logDetails{
|
|
|
|
TopologyID: s.topology.ID,
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, cluster := range s.topology.Clusters {
|
|
|
|
client := s.clients[cluster.Name]
|
|
|
|
|
|
|
|
cfg, err := client.Operator().RaftGetConfiguration(nil)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("could not get raft config for cluster %q: %w", cluster.Name, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var leaderNode string
|
|
|
|
for _, svr := range cfg.Servers {
|
|
|
|
if svr.Leader {
|
|
|
|
leaderNode = strings.TrimSuffix(svr.Node, "-pod")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cd := clusterDetails{
|
|
|
|
Name: cluster.Name,
|
|
|
|
Leader: leaderNode,
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, node := range cluster.Nodes {
|
|
|
|
if node.Disabled {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
var addrs []string
|
|
|
|
for _, addr := range node.Addresses {
|
|
|
|
addrs = append(addrs, addr.Network+"="+addr.IPAddress)
|
|
|
|
}
|
|
|
|
sort.Strings(addrs)
|
|
|
|
|
|
|
|
if node.IsServer() {
|
|
|
|
cd.Apps = append(cd.Apps, appDetail{
|
|
|
|
Type: "server",
|
|
|
|
Container: node.DockerName(),
|
|
|
|
Addresses: addrs,
|
|
|
|
ExposedPort: node.ExposedPort(8500),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, svc := range node.Services {
|
|
|
|
if svc.IsMeshGateway {
|
|
|
|
cd.Apps = append(cd.Apps, appDetail{
|
|
|
|
Type: "mesh-gateway",
|
|
|
|
Container: node.DockerName(),
|
|
|
|
ExposedPort: node.ExposedPort(svc.Port),
|
|
|
|
ExposedEnvoyAdminPort: node.ExposedPort(svc.EnvoyAdminPort),
|
|
|
|
Addresses: addrs,
|
|
|
|
Service: svc.ID.String(),
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
cd.Apps = append(cd.Apps, appDetail{
|
|
|
|
Type: "app",
|
|
|
|
Container: node.DockerName(),
|
|
|
|
ExposedPort: node.ExposedPort(svc.Port),
|
|
|
|
ExposedEnvoyAdminPort: node.ExposedPort(svc.EnvoyAdminPort),
|
|
|
|
Addresses: addrs,
|
|
|
|
Service: svc.ID.String(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
det.Clusters = append(det.Clusters, cd)
|
|
|
|
}
|
|
|
|
|
|
|
|
var buf bytes.Buffer
|
|
|
|
w := tabwriter.NewWriter(&buf, 0, 0, 3, ' ', tabwriter.Debug)
|
|
|
|
|
|
|
|
score := map[string]int{
|
|
|
|
"server": 0,
|
|
|
|
"mesh-gateway": 1,
|
|
|
|
"app": 2,
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, cluster := range det.Clusters {
|
|
|
|
fmt.Fprintf(w, "CLUSTER\tTYPE\tCONTAINER\tNAME\tADDRS\tPORTS\t\n")
|
|
|
|
sort.Slice(cluster.Apps, func(i, j int) bool {
|
|
|
|
a := cluster.Apps[i]
|
|
|
|
b := cluster.Apps[j]
|
|
|
|
|
|
|
|
asc := score[a.Type]
|
|
|
|
bsc := score[b.Type]
|
|
|
|
|
|
|
|
if asc < bsc {
|
|
|
|
return true
|
|
|
|
} else if asc > bsc {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if a.Container < b.Container {
|
|
|
|
return true
|
|
|
|
} else if a.Container > b.Container {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if a.Service < b.Service {
|
|
|
|
return true
|
|
|
|
} else if a.Service > b.Service {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return a.ExposedPort < b.ExposedPort
|
|
|
|
})
|
|
|
|
for _, d := range cluster.Apps {
|
|
|
|
if d.Type == "server" && d.Container == cluster.Leader {
|
|
|
|
d.Type = "leader"
|
|
|
|
}
|
|
|
|
portStr := "app=" + strconv.Itoa(d.ExposedPort)
|
|
|
|
if d.ExposedEnvoyAdminPort > 0 {
|
|
|
|
portStr += " envoy=" + strconv.Itoa(d.ExposedEnvoyAdminPort)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, "%s\t%s\t%s\t%s\t%s\t%s\t\n",
|
|
|
|
cluster.Name,
|
|
|
|
d.Type,
|
|
|
|
d.Container,
|
|
|
|
d.Service,
|
|
|
|
strings.Join(d.Addresses, ", "),
|
|
|
|
portStr,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, "\t\t\t\t\t\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Flush()
|
|
|
|
|
|
|
|
s.logger.Info("CURRENT SPRAWL DETAILS", "details", buf.String())
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type logDetails struct {
|
|
|
|
TopologyID string
|
|
|
|
Clusters []clusterDetails
|
|
|
|
}
|
|
|
|
|
|
|
|
type clusterDetails struct {
|
|
|
|
Name string
|
|
|
|
|
|
|
|
Leader string
|
|
|
|
Apps []appDetail
|
|
|
|
}
|
|
|
|
|
|
|
|
type appDetail struct {
|
|
|
|
Type string // server|mesh-gateway|app
|
|
|
|
Container string
|
|
|
|
Addresses []string
|
|
|
|
ExposedPort int `json:",omitempty"`
|
|
|
|
ExposedEnvoyAdminPort int `json:",omitempty"`
|
|
|
|
// just services
|
|
|
|
Service string `json:",omitempty"`
|
|
|
|
}
|