frontend: BGPmap improvements (#36)

* bgpmap: Compact nexthop info into an edge label

* bgpmap: parse and show non-BGP routes

* bgpmap: Misc tweaks

- Show the protocol name instead of the ASN in edge labels
- Correctly draw only the primary path if there are multiple routes to the first neighbour ASN in a path
- Use a smaller font size for edge labels

* bgpmap_test: update to match new changes

* bgpmap: Split route info on all (non-empty) rta_dest_names values
This commit is contained in:
James Lu 2021-08-31 07:44:14 -07:00 committed by GitHub
parent fbd190628c
commit 1a3c618522
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 56 additions and 27 deletions

View File

@ -4,6 +4,7 @@ import (
"fmt" "fmt"
"html" "html"
"net" "net"
"regexp"
"strings" "strings"
) )
@ -48,8 +49,8 @@ func birdRouteToGraphviz(servers []string, responses []string, target string) st
addEdge := func(src string, dest string, attr string) { addEdge := func(src string, dest string, attr string) {
key := "\"" + html.EscapeString(src) + "\" -> \"" + html.EscapeString(dest) + "\"" key := "\"" + html.EscapeString(src) + "\" -> \"" + html.EscapeString(dest) + "\""
_, present := graph[key] _, present := graph[key]
// Do not remove edge's attributes if it's already present // If there are multiple edges / routes between 2 nodes, only pick the first one
if present && len(attr) == 0 { if present {
return return
} }
graph[key] = attr graph[key] = attr
@ -64,6 +65,12 @@ func birdRouteToGraphviz(servers []string, responses []string, target string) st
} }
graph[key] = attr graph[key] = attr
} }
// 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,shape=diamond]") addPoint("Target: "+target, "[color=red,shape=diamond]")
for serverID, server := range servers { for serverID, server := range servers {
@ -72,24 +79,44 @@ func birdRouteToGraphviz(servers []string, responses []string, target string) st
continue continue
} }
addPoint(server, "[color=blue,shape=box]") addPoint(server, "[color=blue,shape=box]")
// This is the best split point I can find for bird2 routes := routeSplitRe.Split(response, -1)
routes := strings.Split(response, "\tvia ")
routeFound := false targetNodeName := "Target: " + target
var nonBGPRoutes []string
var nonBGPRoutePreferred bool
for routeIndex, route := range routes { for routeIndex, route := range routes {
var routeNexthop string var routeNexthop string
var routeASPath string var routeASPath string
var routePreferred bool = routeIndex > 0 && strings.Contains(routes[routeIndex-1], "*") var routePreferred bool = routeIndex > 0 && strings.Contains(route, "*")
// Have to look at previous slice to determine if route is preferred, due to bad split point selection // 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
var protocolName string
for _, routeParameter := range strings.Split(route, "\n") { for _, routeParameter := range strings.Split(route, "\n") {
if strings.HasPrefix(routeParameter, "\tBGP.next_hop: ") { if strings.HasPrefix(routeParameter, "\tBGP.next_hop: ") {
routeNexthop = strings.TrimPrefix(routeParameter, "\tBGP.next_hop: ") routeNexthop = strings.TrimPrefix(routeParameter, "\tBGP.next_hop: ")
} else if strings.HasPrefix(routeParameter, "\tBGP.as_path: ") { } else if strings.HasPrefix(routeParameter, "\tBGP.as_path: ") {
routeASPath = strings.TrimPrefix(routeParameter, "\tBGP.as_path: ") routeASPath = strings.TrimPrefix(routeParameter, "\tBGP.as_path: ")
} else {
match := protocolNameRe.FindStringSubmatch(routeParameter)
if len(match) >= 2 {
protocolName = match[1]
}
} }
} }
if routePreferred {
protocolName = protocolName + "*"
}
if len(routeASPath) == 0 { if len(routeASPath) == 0 {
// Either this is not a BGP route, or the information is incomplete if routeIndex == 0 {
// The first string split includes the target prefix and isn't a valid route
continue
}
if routePreferred {
nonBGPRoutePreferred = true
}
nonBGPRoutes = append(nonBGPRoutes, protocolName)
continue continue
} }
@ -103,18 +130,15 @@ func birdRouteToGraphviz(servers []string, responses []string, target string) st
// First step starting from originating server // First step starting from originating server
if len(paths) > 0 { if len(paths) > 0 {
if len(routeNexthop) > 0 { attrs := []string{"fontsize=12.0"}
// Edge from originating server to nexthop if routePreferred {
addEdge(server, "Nexthop:\\n"+routeNexthop, (map[bool]string{true: "[color=red]"})[routePreferred]) attrs = append(attrs, "color=red")
// and from nexthop to AS
addEdge("Nexthop:\\n"+routeNexthop, getASNRepresentation(paths[0]), (map[bool]string{true: "[color=red]"})[routePreferred])
addPoint("Nexthop:\\n"+routeNexthop, "[shape=diamond]")
routeFound = true
} else {
// Edge from originating server to AS
addEdge(server, getASNRepresentation(paths[0]), (map[bool]string{true: "[color=red]"})[routePreferred])
routeFound = true
} }
if len(routeNexthop) > 0 {
attrs = append(attrs, fmt.Sprintf("label=\"%s\\n%s\"", protocolName, routeNexthop))
}
formattedAttr := fmt.Sprintf("[%s]", strings.Join(attrs, ","))
addEdge(server, getASNRepresentation(paths[0]), formattedAttr)
} }
// Following steps, edges between AS // Following steps, edges between AS
@ -125,12 +149,19 @@ func birdRouteToGraphviz(servers []string, responses []string, target string) st
addEdge(getASNRepresentation(paths[pathIndex-1]), getASNRepresentation(paths[pathIndex]), (map[bool]string{true: "[color=red]"})[routePreferred]) addEdge(getASNRepresentation(paths[pathIndex-1]), getASNRepresentation(paths[pathIndex]), (map[bool]string{true: "[color=red]"})[routePreferred])
} }
// Last AS to destination // Last AS to destination
addEdge(getASNRepresentation(paths[len(paths)-1]), "Target: "+target, (map[bool]string{true: "[color=red]"})[routePreferred]) addEdge(getASNRepresentation(paths[len(paths)-1]), targetNodeName, (map[bool]string{true: "[color=red]"})[routePreferred])
} }
if !routeFound { if len(nonBGPRoutes) > 0 {
// Cannot find a path starting from this server protocolsForRoute := fmt.Sprintf("label=\"%s\"", strings.Join(nonBGPRoutes, "\\n"))
addEdge(server, "Target: "+target, "[color=gray,label=\"?\"]")
attrs := []string{protocolsForRoute, "fontsize=12.0"}
if nonBGPRoutePreferred {
attrs = append(attrs, "color=red")
}
formattedAttr := fmt.Sprintf("[%s]", strings.Join(attrs, ","))
addEdge(server, targetNodeName, formattedAttr)
} }
} }

View File

@ -45,12 +45,10 @@ func TestBirdRouteToGraphviz(t *testing.T) {
BGP.next_hop: 172.18.0.2` BGP.next_hop: 172.18.0.2`
expectedResult := `digraph { expectedResult := `digraph {
"Nexthop:\n172.18.0.2" -> "AS4242422601" [color=red];
"Nexthop:\n172.18.0.2" [shape=diamond];
"AS4242422601" -> "Target: 192.168.0.1" [color=red];
"Target: 192.168.0.1" [color=red,shape=diamond]; "Target: 192.168.0.1" [color=red,shape=diamond];
"alpha" [color=blue,shape=box]; "alpha" [color=blue,shape=box];
"alpha" -> "Nexthop:\n172.18.0.2" [color=red]; "alpha" -> "AS4242422601" [fontsize=12.0,color=red,label="alpha*\n172.18.0.2"];
"AS4242422601" -> "Target: 192.168.0.1" [color=red];
}` }`
result := birdRouteToGraphviz([]string{ result := birdRouteToGraphviz([]string{