2023-11-02 14:25:48 -05:00
|
|
|
// Copyright (c) HashiCorp, Inc.
|
|
|
|
// SPDX-License-Identifier: BUSL-1.1
|
|
|
|
|
|
|
|
package topology
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"text/tabwriter"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ComputeRelationships will analyze a full topology and generate all of the
|
|
|
|
// downstream/upstream information for all of them.
|
|
|
|
func (t *Topology) ComputeRelationships() []Relationship {
|
|
|
|
var out []Relationship
|
|
|
|
for _, cluster := range t.Clusters {
|
|
|
|
for _, n := range cluster.Nodes {
|
|
|
|
for _, s := range n.Services {
|
|
|
|
for _, u := range s.Upstreams {
|
|
|
|
out = append(out, Relationship{
|
|
|
|
Caller: s,
|
|
|
|
Upstream: u,
|
|
|
|
})
|
|
|
|
}
|
2023-11-02 16:13:16 -05:00
|
|
|
for _, u := range s.ImpliedUpstreams {
|
|
|
|
out = append(out, Relationship{
|
|
|
|
Caller: s,
|
|
|
|
Upstream: u,
|
|
|
|
})
|
|
|
|
}
|
2023-11-02 14:25:48 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
// RenderRelationships will take the output of ComputeRelationships and display
|
|
|
|
// it in tabular form.
|
|
|
|
func RenderRelationships(ships []Relationship) string {
|
|
|
|
var buf bytes.Buffer
|
|
|
|
w := tabwriter.NewWriter(&buf, 0, 0, 3, ' ', tabwriter.Debug)
|
|
|
|
fmt.Fprintf(w, "DOWN\tnode\tservice\tport\tUP\tservice\t\n")
|
|
|
|
for _, r := range ships {
|
2023-11-02 16:13:16 -05:00
|
|
|
suffix := ""
|
|
|
|
if r.Upstream.Implied {
|
|
|
|
suffix = " (implied)"
|
|
|
|
}
|
2023-11-02 14:25:48 -05:00
|
|
|
fmt.Fprintf(w,
|
|
|
|
"%s\t%s\t%s\t%d\t%s\t%s\t\n",
|
|
|
|
r.downCluster(),
|
|
|
|
r.Caller.Node.ID().String(),
|
|
|
|
r.Caller.ID.String(),
|
|
|
|
r.Upstream.LocalPort,
|
|
|
|
r.upCluster(),
|
2023-11-02 16:13:16 -05:00
|
|
|
r.Upstream.ID.String()+suffix,
|
2023-11-02 14:25:48 -05:00
|
|
|
)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, "\t\t\t\t\t\t\n")
|
|
|
|
|
|
|
|
w.Flush()
|
|
|
|
return buf.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
type Relationship struct {
|
|
|
|
Caller *Service
|
|
|
|
Upstream *Upstream
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r Relationship) String() string {
|
2023-11-02 16:13:16 -05:00
|
|
|
suffix := ""
|
|
|
|
if r.Upstream.PortName != "" {
|
|
|
|
suffix = " port " + r.Upstream.PortName
|
|
|
|
}
|
2023-11-02 14:25:48 -05:00
|
|
|
return fmt.Sprintf(
|
2023-11-02 16:13:16 -05:00
|
|
|
"%s on %s in %s via :%d => %s in %s%s",
|
2023-11-02 14:25:48 -05:00
|
|
|
r.Caller.ID.String(),
|
|
|
|
r.Caller.Node.ID().String(),
|
|
|
|
r.downCluster(),
|
|
|
|
r.Upstream.LocalPort,
|
|
|
|
r.Upstream.ID.String(),
|
|
|
|
r.upCluster(),
|
2023-11-02 16:13:16 -05:00
|
|
|
suffix,
|
2023-11-02 14:25:48 -05:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r Relationship) downCluster() string {
|
|
|
|
return r.Caller.Node.Cluster
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r Relationship) upCluster() string {
|
|
|
|
return r.Upstream.Cluster
|
|
|
|
}
|