frontend: refactor bgpmap code to fix #75

This commit is contained in:
Lan Tian 2023-05-05 01:58:05 -07:00
parent 7b0c8c0556
commit 7efa3237a9
No known key found for this signature in database
GPG Key ID: 04E66B6B25A0862B
2 changed files with 173 additions and 134 deletions

View File

@ -8,6 +8,18 @@ import (
"strings" "strings"
) )
// The protocol name for each route (e.g. "ibgp_sea02") is encoded in the form:
//
// unicast [ibgp_sea02 2021-08-27 from fd86:bad:11b7:1::1] * (100/1015) [i]
var protocolNameRe = regexp.MustCompile(`\[(.*?) .*\]`)
// Try to split the output into one chunk for each route.
// Possible values are defined at https://gitlab.nic.cz/labs/bird/-/blob/v2.0.8/nest/rt-attr.c#L81-87
var routeSplitRe = regexp.MustCompile("(unicast|blackhole|unreachable|prohibited)")
var routeViaRe = regexp.MustCompile(`(?m)^\t(via .*?)$`)
var routeASPathRe = regexp.MustCompile(`(?m)^\tBGP\.as_path: (.*?)$`)
func graphvizEscape(s string) string { func graphvizEscape(s string) string {
result, err := json.Marshal(s) result, err := json.Marshal(s)
if err != nil { if err != nil {
@ -17,7 +29,16 @@ func graphvizEscape(s string) string {
} }
} }
func getASNRepresentation(asn string) string { type ASNCache map[string]string
func (cache ASNCache) lookup(asn string) string {
var representation string
cachedValue, cacheOk := cache[asn]
if cacheOk {
return cachedValue
}
if setting.dnsInterface != "" { if setting.dnsInterface != "" {
// get ASN representation using DNS // get ASN representation using DNS
records, err := net.LookupTXT(fmt.Sprintf("AS%s.%s", asn, setting.dnsInterface)) records, err := net.LookupTXT(fmt.Sprintf("AS%s.%s", asn, setting.dnsInterface))
@ -26,11 +47,9 @@ func getASNRepresentation(asn string) string {
if resultSplit := strings.Split(result, " | "); len(resultSplit) > 1 { if resultSplit := strings.Split(result, " | "); len(resultSplit) > 1 {
result = strings.Join(resultSplit[1:], "\n") result = strings.Join(resultSplit[1:], "\n")
} }
return fmt.Sprintf("AS%s\n%s", asn, result) representation = fmt.Sprintf("AS%s\n%s", asn, result)
} }
} } else if setting.whoisServer != "" {
if setting.whoisServer != "" {
// get ASN representation using WHOIS // get ASN representation using WHOIS
if setting.bgpmapInfo == "" { if setting.bgpmapInfo == "" {
setting.bgpmapInfo = "asn,as-name,ASName,descr" setting.bgpmapInfo = "asn,as-name,ASName,descr"
@ -68,167 +87,185 @@ func getASNRepresentation(asn string) string {
} }
} }
if len(result) > 0 { if len(result) > 0 {
return fmt.Sprintf("%s", strings.Join(result, "\n")) representation = strings.Join(result, "\n")
} }
} }
} else {
representation = fmt.Sprintf("AS%s", asn)
} }
return fmt.Sprintf("AS%s", asn)
cache[asn] = representation
return representation
} }
func birdRouteToGraphviz(servers []string, responses []string, target string) string { func birdRouteToGraphviz(servers []string, responses []string, targetName string) string {
graph := make(map[string](map[string]string)) asnCache := make(ASNCache)
// Helper to add an edge graph := makeRouteGraph()
addEdge := func(src string, dest string, attrKey string, attrValue string) {
key := graphvizEscape(src) + " -> " + graphvizEscape(dest)
_, present := graph[key]
if !present {
graph[key] = map[string]string{}
}
if attrKey != "" {
graph[key][attrKey] = attrValue
}
}
// Helper to set attribute for a point in graph
addPoint := func(name string, attrKey string, attrValue string) {
key := graphvizEscape(name)
_, present := graph[key]
if !present {
graph[key] = map[string]string{}
}
if attrKey != "" {
graph[key][attrKey] = attrValue
}
}
// The protocol name for each route (e.g. "ibgp_sea02") is encoded in the form:
// unicast [ibgp_sea02 2021-08-27 from fd86:bad:11b7:1::1] * (100/1015) [i]
protocolNameRe := regexp.MustCompile(`\[(.*?) .*\]`)
// Try to split the output into one chunk for each route.
// Possible values are defined at https://gitlab.nic.cz/labs/bird/-/blob/v2.0.8/nest/rt-attr.c#L81-87
routeSplitRe := regexp.MustCompile("(unicast|blackhole|unreachable|prohibited)")
addPoint("Target: "+target, "color", "red") makeEdgeAttrs := func(preferred bool) RouteAttrs {
addPoint("Target: "+target, "shape", "diamond") result := RouteAttrs{
"fontsize": "12.0",
}
if preferred {
result["color"] = "red"
}
return result
}
makePointAttrs := func(preferred bool) RouteAttrs {
result := RouteAttrs{}
if preferred {
result["color"] = "red"
}
return result
}
target := "Target: " + targetName
graph.AddPoint(target, RouteAttrs{"color": "red", "shape": "diamond"})
for serverID, server := range servers { for serverID, server := range servers {
response := responses[serverID] response := responses[serverID]
if len(response) == 0 { if len(response) == 0 {
continue continue
} }
addPoint(server, "color", "blue") graph.AddPoint(server, RouteAttrs{"color": "blue", "shape": "box"})
addPoint(server, "shape", "box")
routes := routeSplitRe.Split(response, -1) routes := routeSplitRe.Split(response, -1)
targetNodeName := "Target: " + target
var nonBGPRoutes []string
var nonBGPRoutePreferred bool
for routeIndex, route := range routes { for routeIndex, route := range routes {
var routeNexthop string if routeIndex == 0 {
var routeASPath string continue
var routePreferred bool = routeIndex > 0 && strings.Contains(route, "*") }
var via string
var paths []string
var routePreferred bool = strings.Contains(route, "*")
// Track non-BGP routes in the output by their protocol name, but draw them altogether in one line // Track non-BGP routes in the output by their protocol name, but draw them altogether in one line
// so that there are no conflicts in the edge label // so that there are no conflicts in the edge label
var protocolName string var protocolName string
for _, routeParameter := range strings.Split(route, "\n") { if match := routeViaRe.FindStringSubmatch(route); len(match) >= 2 {
if strings.HasPrefix(routeParameter, "\tBGP.next_hop: ") { via = strings.TrimSpace(match[1])
routeNexthop = strings.TrimPrefix(routeParameter, "\tBGP.next_hop: ") }
} else if strings.HasPrefix(routeParameter, "\tBGP.as_path: ") {
routeASPath = strings.TrimPrefix(routeParameter, "\tBGP.as_path: ") if match := routeASPathRe.FindStringSubmatch(route); len(match) >= 2 {
} else { pathString := strings.TrimSpace(match[1])
match := protocolNameRe.FindStringSubmatch(routeParameter) if len(pathString) > 0 {
if len(match) >= 2 { paths = strings.Split(strings.TrimSpace(match[1]), " ")
protocolName = match[1] for i := range paths {
paths[i] = strings.TrimPrefix(paths[i], "(")
paths[i] = strings.TrimSuffix(paths[i], ")")
} }
} }
} }
if routePreferred {
protocolName = protocolName + "*" if match := protocolNameRe.FindStringSubmatch(route); len(match) >= 2 {
} protocolName = strings.TrimSpace(match[1])
if len(routeASPath) == 0 {
if routeIndex == 0 {
// The first string split includes the target prefix and isn't a valid route
continue
}
if routePreferred { if routePreferred {
nonBGPRoutePreferred = true protocolName = protocolName + "*"
} }
nonBGPRoutes = append(nonBGPRoutes, protocolName) }
if len(paths) == 0 {
graph.AddEdge(server, target, strings.TrimSpace(protocolName+"\n"+via), makeEdgeAttrs(routePreferred))
continue continue
} }
// Connect each node on AS path // Edges between AS
paths := strings.Split(strings.TrimSpace(routeASPath), " ") for i := range paths {
var src string
for pathIndex := range paths { if i == 0 {
paths[pathIndex] = strings.TrimPrefix(paths[pathIndex], "(") src = server
paths[pathIndex] = strings.TrimSuffix(paths[pathIndex], ")")
}
// First step starting from originating server
if len(paths) > 0 {
edgeTarget := getASNRepresentation(paths[0])
addEdge(server, edgeTarget, "fontsize", "12.0")
if routePreferred {
addEdge(server, edgeTarget, "color", "red")
// Only set color for next step, origin color is set to blue above
addPoint(edgeTarget, "color", "red")
}
if len(routeNexthop) > 0 {
addEdge(server, edgeTarget, "label", protocolName + "\n" + routeNexthop)
}
}
// Following steps, edges between AS
for pathIndex := range paths {
if pathIndex == 0 {
continue
}
if routePreferred {
addEdge(getASNRepresentation(paths[pathIndex-1]), getASNRepresentation(paths[pathIndex]), "color", "red")
// Only set color for next step, origin color is set to blue above
addPoint(getASNRepresentation(paths[pathIndex]), "color", "red")
} else { } else {
addEdge(getASNRepresentation(paths[pathIndex-1]), getASNRepresentation(paths[pathIndex]), "", "") src = asnCache.lookup(paths[i-1])
} }
dst := asnCache.lookup(paths[i])
graph.AddEdge(src, dst, strings.TrimSpace(protocolName+"\n"+via), makeEdgeAttrs(routePreferred))
// Only set color for next step, origin color is set to blue above
graph.AddPoint(dst, makePointAttrs(routePreferred))
} }
// Last AS to destination // Last AS to destination
if routePreferred { src := asnCache.lookup(paths[len(paths)-1])
addEdge(getASNRepresentation(paths[len(paths)-1]), targetNodeName, "color", "red") graph.AddEdge(src, target, "", makeEdgeAttrs(routePreferred))
} else {
addEdge(getASNRepresentation(paths[len(paths)-1]), targetNodeName, "", "")
}
}
if len(nonBGPRoutes) > 0 {
addEdge(server, targetNodeName, "label", strings.Join(nonBGPRoutes, "\n"))
addEdge(server, targetNodeName, "fontsize", "12.0")
if nonBGPRoutePreferred {
addEdge(server, targetNodeName, "color", "red")
}
} }
} }
// Combine all graphviz commands return graph.ToGraphviz()
var result string }
for edge, attr := range graph {
result += edge; type RouteGraph struct {
if len(attr) != 0 { points map[string]RouteAttrs
result += " [" edges map[RouteEdge]RouteAttrs
isFirst := true }
for k, v := range attr { type RouteEdge struct {
if isFirst { src string
isFirst = false dest string
} else { label string
result += "," }
} type RouteAttrs map[string]string
result += graphvizEscape(k) + "=" + graphvizEscape(v) + "";
} func attrsToString(attrs RouteAttrs) string {
result += "]" if len(attrs) == 0 {
return ""
}
result := ""
isFirst := true
for k, v := range attrs {
if isFirst {
isFirst = false
} else {
result += ","
} }
result += ";\n" result += graphvizEscape(k) + "=" + graphvizEscape(v) + ""
}
return "[" + result + "]"
}
func makeRouteGraph() RouteGraph {
return RouteGraph{
points: make(map[string]RouteAttrs),
edges: make(map[RouteEdge]RouteAttrs),
}
}
func (graph *RouteGraph) AddEdge(src string, dest string, label string, attrs RouteAttrs) {
// Add edges with same src/dest separately, multiple edges with same src/dest could exist
edge := RouteEdge{
src: src,
dest: dest,
label: label,
}
_, exists := graph.edges[edge]
if !exists {
graph.edges[edge] = make(RouteAttrs)
}
for k, v := range attrs {
graph.edges[edge][k] = v
}
}
func (graph *RouteGraph) AddPoint(name string, attrs RouteAttrs) {
graph.points[name] = attrs
}
func (graph *RouteGraph) ToGraphviz() string {
var result string
for name, attrs := range graph.points {
result += fmt.Sprintf("%s %s;\n", graphvizEscape(name), attrsToString(attrs))
}
for edge, attrs := range graph.edges {
attrsCopy := attrs
if attrsCopy == nil {
attrsCopy = make(RouteAttrs)
}
if len(edge.label) > 0 {
attrsCopy["label"] = edge.label
}
result += fmt.Sprintf("%s -> %s %s;\n", graphvizEscape(edge.src), graphvizEscape(edge.dest), attrsToString(attrsCopy))
} }
return "digraph {\n" + result + "}\n" return "digraph {\n" + result + "}\n"
} }

View File

@ -20,7 +20,8 @@ func TestGetASNRepresentationDNS(t *testing.T) {
setting.dnsInterface = "asn.cymru.com" setting.dnsInterface = "asn.cymru.com"
setting.whoisServer = "" setting.whoisServer = ""
result := getASNRepresentation("6939") cache := make(ASNCache)
result := cache.lookup("6939")
if !strings.Contains(result, "HURRICANE") { if !strings.Contains(result, "HURRICANE") {
t.Errorf("Lookup AS6939 failed, got %s", result) t.Errorf("Lookup AS6939 failed, got %s", result)
} }
@ -31,7 +32,8 @@ func TestGetASNRepresentationWhois(t *testing.T) {
setting.dnsInterface = "" setting.dnsInterface = ""
setting.whoisServer = "whois.arin.net" setting.whoisServer = "whois.arin.net"
result := getASNRepresentation("6939") cache := make(ASNCache)
result := cache.lookup("6939")
if !strings.Contains(result, "HURRICANE") { if !strings.Contains(result, "HURRICANE") {
t.Errorf("Lookup AS6939 failed, got %s", result) t.Errorf("Lookup AS6939 failed, got %s", result)
} }
@ -40,7 +42,8 @@ func TestGetASNRepresentationWhois(t *testing.T) {
func TestGetASNRepresentationFallback(t *testing.T) { func TestGetASNRepresentationFallback(t *testing.T) {
setting.dnsInterface = "" setting.dnsInterface = ""
setting.whoisServer = "" setting.whoisServer = ""
result := getASNRepresentation("6939") cache := make(ASNCache)
result := cache.lookup("6939")
if result != "AS6939" { if result != "AS6939" {
t.Errorf("Lookup AS6939 failed, got %s", result) t.Errorf("Lookup AS6939 failed, got %s", result)
} }
@ -73,7 +76,6 @@ func TestBirdRouteToGraphviz(t *testing.T) {
fakeResult, fakeResult,
}, "192.168.0.1") }, "192.168.0.1")
for _, line := range expectedLinesInResult { for _, line := range expectedLinesInResult {
if !strings.Contains(result, line) { if !strings.Contains(result, line) {
t.Errorf("Expected line in result not found: %s", line) t.Errorf("Expected line in result not found: %s", line)