mirror of
https://github.com/weaveworks/scope.git
synced 2026-03-02 17:50:39 +00:00
Pass 'id' through to the create function and expect that the result Node has that ID. Extract a function newPseudoNode for common calls.
102 lines
3.1 KiB
Go
102 lines
3.1 KiB
Go
package render
|
|
|
|
import (
|
|
"net"
|
|
"strings"
|
|
|
|
"github.com/weaveworks/scope/probe/endpoint"
|
|
"github.com/weaveworks/scope/report"
|
|
)
|
|
|
|
// Constants are used in the tests.
|
|
const (
|
|
TheInternetID = "theinternet"
|
|
IncomingInternetID = "in-" + TheInternetID
|
|
OutgoingInternetID = "out-" + TheInternetID
|
|
)
|
|
|
|
// MakePseudoNodeID joins the parts of an id into the id of a pseudonode
|
|
func MakePseudoNodeID(parts ...string) string {
|
|
return strings.Join(append([]string{"pseudo"}, parts...), ":")
|
|
}
|
|
|
|
// MakeGroupNodeTopology joins the parts of a group topology into the topology of a group node
|
|
func MakeGroupNodeTopology(originalTopology, key string) string {
|
|
return strings.Join([]string{"group", originalTopology, key}, ":")
|
|
}
|
|
|
|
// NewDerivedNode makes a node based on node, but with a new ID
|
|
func NewDerivedNode(id string, node report.Node) report.Node {
|
|
return report.MakeNode(id).WithChildren(node.Children.Add(node))
|
|
}
|
|
|
|
// NewDerivedPseudoNode makes a new pseudo node with the node as a child
|
|
func NewDerivedPseudoNode(id string, node report.Node) report.Node {
|
|
output := NewDerivedNode(id, node).WithTopology(Pseudo)
|
|
return output
|
|
}
|
|
|
|
func newPseudoNode(id string) report.Node {
|
|
return report.MakeNode(id).WithTopology(Pseudo)
|
|
}
|
|
|
|
func pseudoNodeID(n report.Node, local report.Networks) (string, bool) {
|
|
_, addr, _, ok := report.ParseEndpointNodeID(n.ID)
|
|
if !ok {
|
|
return "", false
|
|
}
|
|
|
|
if id, ok := externalNodeID(n, addr, local); ok {
|
|
return id, ok
|
|
}
|
|
|
|
// due to https://github.com/weaveworks/scope/issues/1323 we are dropping
|
|
// all non-external pseudo nodes for now.
|
|
return "", false
|
|
}
|
|
|
|
// figure out if a node should be considered external and returns an ID which can be used to create a pseudo node
|
|
func externalNodeID(n report.Node, addr string, local report.Networks) (string, bool) {
|
|
// First, check if it's a known service and emit a a specific node if it
|
|
// is. This needs to be done before checking IPs since known services can
|
|
// live in the same network, see https://github.com/weaveworks/scope/issues/2163
|
|
if hostname, found := DNSFirstMatch(n, isKnownService); found {
|
|
return ServiceNodeIDPrefix + hostname, true
|
|
}
|
|
|
|
// If the dstNodeAddr is not in a network local to this report, we emit an
|
|
// internet pseudoNode
|
|
if ip := net.ParseIP(addr); ip != nil && !local.Contains(ip) {
|
|
// emit one internet node for incoming, one for outgoing
|
|
if len(n.Adjacency) > 0 {
|
|
return IncomingInternetID, true
|
|
}
|
|
return OutgoingInternetID, true
|
|
}
|
|
|
|
// The node is not external
|
|
return "", false
|
|
}
|
|
|
|
// DNSFirstMatch returns the first DNS name where match() returns
|
|
// true, from a prioritized list of snooped and reverse-resolved DNS
|
|
// names associated with node n.
|
|
func DNSFirstMatch(n report.Node, match func(name string) bool) (string, bool) {
|
|
// we rely on Sets being sorted, to make selection for display more
|
|
// deterministic
|
|
// prioritize snooped names
|
|
snoopedNames, _ := n.Sets.Lookup(endpoint.SnoopedDNSNames)
|
|
for _, hostname := range snoopedNames {
|
|
if match(hostname) {
|
|
return hostname, true
|
|
}
|
|
}
|
|
reverseNames, _ := n.Sets.Lookup(endpoint.ReverseDNSNames)
|
|
for _, hostname := range reverseNames {
|
|
if match(hostname) {
|
|
return hostname, true
|
|
}
|
|
}
|
|
return "", false
|
|
}
|