frontend: refactor bgpmap and fix node colors (#67)

* frontend: refactor bgpmap and fix node colors

* frontend: alternative way to test bgpmap
This commit is contained in:
Yuhui Xu 2022-12-07 16:30:19 -06:00 committed by GitHub
parent 335ad40634
commit 9e17b116f1
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 91 additions and 41 deletions

View File

@ -1,13 +1,22 @@
package main package main
import ( import (
"encoding/json"
"fmt" "fmt"
"html"
"net" "net"
"regexp" "regexp"
"strings" "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 { func getASNRepresentation(asn string) string {
if setting.dnsInterface != "" { if setting.dnsInterface != "" {
// get ASN representation using DNS // get ASN representation using DNS
@ -15,9 +24,9 @@ func getASNRepresentation(asn string) string {
if err == nil { if err == nil {
result := strings.Join(records, " ") result := strings.Join(records, " ")
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) return fmt.Sprintf("AS%s\n%s", asn, result)
} }
} }
@ -67,26 +76,28 @@ func getASNRepresentation(asn string) string {
} }
func birdRouteToGraphviz(servers []string, responses []string, target string) string { func birdRouteToGraphviz(servers []string, responses []string, target string) string {
graph := make(map[string]string) graph := make(map[string](map[string]string))
// Helper to add an edge // Helper to add an edge
addEdge := func(src string, dest string, attr string) { addEdge := func(src string, dest string, attrKey string, attrValue string) {
key := "\"" + html.EscapeString(src) + "\" -> \"" + html.EscapeString(dest) + "\"" key := graphvizEscape(src) + " -> " + graphvizEscape(dest)
_, present := graph[key] _, present := graph[key]
// If there are multiple edges / routes between 2 nodes, only pick the first one if !present {
if present { graph[key] = map[string]string{}
return }
if attrKey != "" {
graph[key][attrKey] = attrValue
} }
graph[key] = attr
} }
// Helper to set attribute for a point in graph // Helper to set attribute for a point in graph
addPoint := func(name string, attr string) { addPoint := func(name string, attrKey string, attrValue string) {
key := "\"" + html.EscapeString(name) + "\"" key := graphvizEscape(name)
_, present := graph[key] _, present := graph[key]
// Do not remove point's attributes if it's already present if !present {
if present && len(attr) == 0 { graph[key] = map[string]string{}
return }
if attrKey != "" {
graph[key][attrKey] = attrValue
} }
graph[key] = attr
} }
// The protocol name for each route (e.g. "ibgp_sea02") is encoded in the form: // 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] // unicast [ibgp_sea02 2021-08-27 from fd86:bad:11b7:1::1] * (100/1015) [i]
@ -95,13 +106,16 @@ func birdRouteToGraphviz(servers []string, responses []string, target string) st
// Possible values are defined at https://gitlab.nic.cz/labs/bird/-/blob/v2.0.8/nest/rt-attr.c#L81-87 // 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)") routeSplitRe := regexp.MustCompile("(unicast|blackhole|unreachable|prohibited)")
addPoint("Target: "+target, "[color=red,shape=diamond]") addPoint("Target: "+target, "color", "red")
addPoint("Target: "+target, "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,shape=box]") addPoint(server, "color", "blue")
addPoint(server, "shape", "box")
routes := routeSplitRe.Split(response, -1) routes := routeSplitRe.Split(response, -1)
targetNodeName := "Target: " + target targetNodeName := "Target: " + target
@ -153,15 +167,16 @@ 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 {
attrs := []string{"fontsize=12.0"} edgeTarget := getASNRepresentation(paths[0])
addEdge(server, edgeTarget, "fontsize", "12.0")
if routePreferred { if routePreferred {
attrs = append(attrs, "color=red") 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 { if len(routeNexthop) > 0 {
attrs = append(attrs, fmt.Sprintf("label=\"%s\\n%s\"", protocolName, routeNexthop)) addEdge(server, edgeTarget, "label", protocolName + "\n" + routeNexthop)
} }
formattedAttr := fmt.Sprintf("[%s]", strings.Join(attrs, ","))
addEdge(server, getASNRepresentation(paths[0]), formattedAttr)
} }
// Following steps, edges between AS // Following steps, edges between AS
@ -169,29 +184,51 @@ func birdRouteToGraphviz(servers []string, responses []string, target string) st
if pathIndex == 0 { if pathIndex == 0 {
continue continue
} }
addEdge(getASNRepresentation(paths[pathIndex-1]), getASNRepresentation(paths[pathIndex]), (map[bool]string{true: "[color=red]"})[routePreferred]) 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 // Last AS to destination
addEdge(getASNRepresentation(paths[len(paths)-1]), targetNodeName, (map[bool]string{true: "[color=red]"})[routePreferred]) if routePreferred {
addEdge(getASNRepresentation(paths[len(paths)-1]), targetNodeName, "color", "red")
} else {
addEdge(getASNRepresentation(paths[len(paths)-1]), targetNodeName, "", "")
}
} }
if len(nonBGPRoutes) > 0 { if len(nonBGPRoutes) > 0 {
protocolsForRoute := fmt.Sprintf("label=\"%s\"", strings.Join(nonBGPRoutes, "\\n")) addEdge(server, targetNodeName, "label", strings.Join(nonBGPRoutes, "\n"))
addEdge(server, targetNodeName, "fontsize", "12.0")
attrs := []string{protocolsForRoute, "fontsize=12.0"}
if nonBGPRoutePreferred { if nonBGPRoutePreferred {
attrs = append(attrs, "color=red") addEdge(server, targetNodeName, "color", "red")
} }
formattedAttr := fmt.Sprintf("[%s]", strings.Join(attrs, ","))
addEdge(server, targetNodeName, formattedAttr)
} }
} }
// Combine all graphviz commands // Combine all graphviz commands
var result string var result string
for edge, attr := range graph { for edge, attr := range graph {
result += edge + " " + attr + ";\n" 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" return "digraph {\n" + result + "}\n"
} }

View File

@ -5,6 +5,16 @@ import (
"testing" "testing"
) )
func contains(s []string, str string) bool {
for _, v := range s {
if v == str {
return true
}
}
return false
}
func TestGetASNRepresentationDNS(t *testing.T) { func TestGetASNRepresentationDNS(t *testing.T) {
checkNetwork(t) checkNetwork(t)
@ -36,6 +46,7 @@ func TestGetASNRepresentationFallback(t *testing.T) {
} }
} }
// Broken due to random order of attributes
func TestBirdRouteToGraphviz(t *testing.T) { func TestBirdRouteToGraphviz(t *testing.T) {
setting.dnsInterface = "" setting.dnsInterface = ""
@ -48,12 +59,13 @@ func TestBirdRouteToGraphviz(t *testing.T) {
BGP.as_path: 4242422601 BGP.as_path: 4242422601
BGP.next_hop: 172.18.0.2` BGP.next_hop: 172.18.0.2`
expectedResult := `digraph { expectedLinesInResult := []string{
"Target: 192.168.0.1" [color=red,shape=diamond]; `"AS4242422601" [`,
"alpha" [color=blue,shape=box]; `"AS4242422601" -> "Target: 192.168.0.1" [`,
"alpha" -> "AS4242422601" [fontsize=12.0,color=red,label="alpha*\n172.18.0.2"]; `"Target: 192.168.0.1" [`,
"AS4242422601" -> "Target: 192.168.0.1" [color=red]; `"alpha" [`,
}` `"alpha" -> "AS4242422601" [`,
}
result := birdRouteToGraphviz([]string{ result := birdRouteToGraphviz([]string{
"alpha", "alpha",
@ -61,9 +73,10 @@ func TestBirdRouteToGraphviz(t *testing.T) {
fakeResult, fakeResult,
}, "192.168.0.1") }, "192.168.0.1")
for _, line := range strings.Split(result, "\n") {
if !strings.Contains(expectedResult, line) { for _, line := range expectedLinesInResult {
t.Errorf("Unexpected line in result: %s", line) if !strings.Contains(result, line) {
t.Errorf("Expected line in result not found: %s", line)
} }
} }
} }