bird-lg-go/frontend/bgpmap.go
Yuhui Xu 9e17b116f1
frontend: refactor bgpmap and fix node colors (#67)
* frontend: refactor bgpmap and fix node colors

* frontend: alternative way to test bgpmap
2022-12-07 16:30:19 -06:00

235 lines
6.7 KiB
Go

package main
import (
"encoding/json"
"fmt"
"net"
"regexp"
"strings"
)
func graphvizEscape(s string) string {
result, err := json.Marshal(s)
if err != nil {
return err.Error()
} else {
return string(result)
}
}
func getASNRepresentation(asn string) string {
if setting.dnsInterface != "" {
// get ASN representation using DNS
records, err := net.LookupTXT(fmt.Sprintf("AS%s.%s", asn, setting.dnsInterface))
if err == nil {
result := strings.Join(records, " ")
if resultSplit := strings.Split(result, " | "); len(resultSplit) > 1 {
result = strings.Join(resultSplit[1:], "\n")
}
return fmt.Sprintf("AS%s\n%s", asn, result)
}
}
if setting.whoisServer != "" {
// get ASN representation using WHOIS
if setting.bgpmapInfo == "" {
setting.bgpmapInfo = "asn,as-name,ASName,descr"
}
records := whois(fmt.Sprintf("AS%s", asn))
if records != "" {
recordsSplit := strings.Split(records, "\n")
var result []string
for _, title := range strings.Split(setting.bgpmapInfo, ",") {
if title == "asn" {
result = append(result, "AS"+asn)
}
}
for _, title := range strings.Split(setting.bgpmapInfo, ",") {
allow_multiline := false
if title[0] == ':' && len(title) >= 2 {
title = title[1:]
allow_multiline = true
}
for _, line := range recordsSplit {
if len(line) == 0 || line[0] == '%' || !strings.Contains(line, ":") {
continue
}
linearr := strings.SplitN(line, ":", 2)
line_title := linearr[0]
content := strings.TrimSpace(linearr[1])
if line_title != title {
continue
}
result = append(result, content)
if !allow_multiline {
break
}
}
}
if len(result) > 0 {
return fmt.Sprintf("%s", strings.Join(result, "\n"))
}
}
}
return fmt.Sprintf("AS%s", asn)
}
func birdRouteToGraphviz(servers []string, responses []string, target string) string {
graph := make(map[string](map[string]string))
// Helper to add an edge
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")
addPoint("Target: "+target, "shape", "diamond")
for serverID, server := range servers {
response := responses[serverID]
if len(response) == 0 {
continue
}
addPoint(server, "color", "blue")
addPoint(server, "shape", "box")
routes := routeSplitRe.Split(response, -1)
targetNodeName := "Target: " + target
var nonBGPRoutes []string
var nonBGPRoutePreferred bool
for routeIndex, route := range routes {
var routeNexthop string
var routeASPath string
var routePreferred bool = routeIndex > 0 && strings.Contains(route, "*")
// 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") {
if strings.HasPrefix(routeParameter, "\tBGP.next_hop: ") {
routeNexthop = strings.TrimPrefix(routeParameter, "\tBGP.next_hop: ")
} else if strings.HasPrefix(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 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
}
// Connect each node on AS path
paths := strings.Split(strings.TrimSpace(routeASPath), " ")
for pathIndex := range paths {
paths[pathIndex] = strings.TrimPrefix(paths[pathIndex], "(")
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 {
addEdge(getASNRepresentation(paths[pathIndex-1]), getASNRepresentation(paths[pathIndex]), "", "")
}
}
// Last AS to destination
if routePreferred {
addEdge(getASNRepresentation(paths[len(paths)-1]), targetNodeName, "color", "red")
} 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
var result string
for edge, attr := range graph {
result += edge;
if len(attr) != 0 {
result += " ["
isFirst := true
for k, v := range attr {
if isFirst {
isFirst = false
} else {
result += ","
}
result += graphvizEscape(k) + "=" + graphvizEscape(v) + "";
}
result += "]"
}
result += ";\n"
}
return "digraph {\n" + result + "}\n"
}