General code cleanup

This commit is contained in:
Lan Tian 2020-03-27 12:03:48 +08:00
parent 0c06a85a93
commit 8f581d378f
No known key found for this signature in database
GPG Key ID: 27F31700E751EC22
4 changed files with 156 additions and 90 deletions

BIN
.DS_Store vendored

Binary file not shown.

19
.gitignore vendored Normal file
View File

@ -0,0 +1,19 @@
# Binaries for programs and plugins
*.exe
*.exe~
*.dll
*.so
*.dylib
# Test binary, built with `go test -c`
*.test
# Output of the go coverage tool, specifically when used with LiteIDE
*.out
# Dependency directories (remove the comment below to include it)
# vendor/
.DS_Store
frontend/frontend
proxy/proxy

View File

@ -1,20 +1,42 @@
package main package main
import ( import (
"sort"
"strings" "strings"
) )
func birdRouteToGraphviz(servers []string, responses []string, target string) string { func birdRouteToGraphviz(servers []string, responses []string, target string) string {
var edges string graph := make(map[string]string)
edges += "\"Target: " + target + "\" [color=red,shape=diamond];\n" // Helper to add an edge
addEdge := func(src string, dest string, attr string) {
key := "\"" + src + "\" -> \"" + dest + "\""
_, present := graph[key]
// Do not remove edge's attributes if it's already present
if present && len(attr) == 0 {
return
}
graph[key] = attr
}
// Helper to set attribute for a point in graph
addPoint := func(name string, attr string) {
key := "\"" + name + "\""
_, present := graph[key]
// Do not remove point's attributes if it's already present
if present && len(attr) == 0 {
return
}
graph[key] = attr
}
addPoint("Target: "+target, "[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
} }
edges += "\"" + server + "\" [color=blue,shape=box];\n" addPoint(server, "[color=blue,shape=box]")
// This is the best split point I can find for bird2
routes := strings.Split(response, "\tvia ") routes := strings.Split(response, "\tvia ")
routeFound := false
for routeIndex, route := range routes { for routeIndex, route := range routes {
var routeNexthop string var routeNexthop string
var routeASPath string var routeASPath string
@ -28,7 +50,8 @@ func birdRouteToGraphviz(servers []string, responses []string, target string) st
routeASPath = strings.TrimPrefix(routeParameter, "\tBGP.as_path: ") routeASPath = strings.TrimPrefix(routeParameter, "\tBGP.as_path: ")
} }
} }
if len(routeNexthop) == 0 || len(routeASPath) == 0 { if len(routeASPath) == 0 {
// Either this is not a BGP route, or the information is incomplete
continue continue
} }
@ -39,13 +62,15 @@ func birdRouteToGraphviz(servers []string, responses []string, target string) st
if len(paths) > 0 { if len(paths) > 0 {
if len(routeNexthop) > 0 { if len(routeNexthop) > 0 {
// Edge from originating server to nexthop // Edge from originating server to nexthop
edges += "\"" + server + "\" -> \"Nexthop:\\n" + routeNexthop + "\"" + (map[bool]string{true: " [color=red]"})[routePreferred] + ";\n" addEdge(server, "Nexthop:\\n"+routeNexthop, (map[bool]string{true: "[color=red]"})[routePreferred])
// and from nexthop to AS // and from nexthop to AS
edges += "\"Nexthop:\\n" + routeNexthop + "\" -> \"AS" + paths[0] + "\"" + (map[bool]string{true: " [color=red]"})[routePreferred] + ";\n" addEdge("Nexthop:\\n"+routeNexthop, "AS"+paths[0], (map[bool]string{true: "[color=red]"})[routePreferred])
edges += "\"Nexthop:\\n" + routeNexthop + "\" [shape=diamond];\n" addPoint("Nexthop:\\n"+routeNexthop, "[shape=diamond]")
routeFound = true
} else { } else {
// Edge from originating server to AS // Edge from originating server to AS
edges += "\"" + server + "\" -> \"AS" + paths[0] + "\"" + (map[bool]string{true: " [color=red]"})[routePreferred] + ";\n" addEdge(server, "AS"+paths[0], (map[bool]string{true: "[color=red]"})[routePreferred])
routeFound = true
} }
} }
@ -54,25 +79,22 @@ func birdRouteToGraphviz(servers []string, responses []string, target string) st
if pathIndex == 0 { if pathIndex == 0 {
continue continue
} }
edges += "\"AS" + paths[pathIndex-1] + "\" -> \"AS" + paths[pathIndex] + "\"" + (map[bool]string{true: " [color=red]"})[routePreferred] + ";\n" addEdge("AS"+paths[pathIndex-1], "AS"+paths[pathIndex], (map[bool]string{true: "[color=red]"})[routePreferred])
} }
// Last AS to destination // Last AS to destination
edges += "\"AS" + paths[len(paths)-1] + "\" -> \"Target: " + target + "\"" + (map[bool]string{true: " [color=red]"})[routePreferred] + ";\n" addEdge("AS"+paths[len(paths)-1], "Target: "+target, (map[bool]string{true: "[color=red]"})[routePreferred])
} }
if !strings.Contains(edges, "\""+server+"\" ->") {
// Cannot get path information from bird if !routeFound {
edges += "\"" + server + "\" -> \"Target: " + target + "\" [color=gray,label=\"?\"]" // Cannot find a path starting from this server
} addEdge(server, "Target: "+target, "[color=gray,label=\"?\"?]")
}
// Deduplication of edges: sort, then remove if current entry is prefix of next entry
var result string
edgesSorted := strings.Split(edges, ";\n")
sort.Strings(edgesSorted)
for edgeIndex, edge := range edgesSorted {
if edgeIndex >= len(edgesSorted)-1 || !strings.HasPrefix(edgesSorted[edgeIndex+1], edge) {
result += edge + ";\n"
} }
} }
// Combine all graphviz commands
var result string
for edge, attr := range graph {
result += edge + " " + attr + ";\n"
}
return "digraph {\n" + result + "}\n" return "digraph {\n" + result + "}\n"
} }

View File

@ -1,6 +1,7 @@
package main package main
import ( import (
"fmt"
"html" "html"
"net/http" "net/http"
"strings" "strings"
@ -17,65 +18,91 @@ func webHandlerWhois(w http.ResponseWriter, r *http.Request) {
templateFooter(w) templateFooter(w)
} }
func webBackendCommunicator(w http.ResponseWriter, r *http.Request, endpoint string, command string) { func webBackendCommunicator(endpoint string, command string) func(w http.ResponseWriter, r *http.Request) {
split := strings.Split(r.URL.Path[1:], "/") backendCommandPrimitive, commandPresent := (map[string]string{
urlCommands := strings.Join(split[3:], "/")
command = (map[string]string{
"summary": "show protocols", "summary": "show protocols",
"detail": "show protocols all " + urlCommands, "detail": "show protocols all %s",
"route": "show route for " + urlCommands, "route": "show route for %s",
"route_all": "show route for " + urlCommands + " all", "route_all": "show route for %s all",
"route_where": "show route where net ~ [ " + urlCommands + " ]", "route_where": "show route where net ~ [ %s ]",
"route_where_all": "show route where net ~ [ " + urlCommands + " ] all", "route_where_all": "show route where net ~ [ %s ] all",
"traceroute": urlCommands, "traceroute": "%s",
})[command] })[command]
templateHeader(w, r, "Bird-lg Go - "+html.EscapeString(endpoint+" "+command)) if !commandPresent {
panic("invalid command: " + command)
var servers []string = strings.Split(split[2], "+")
var responses []string = batchRequest(servers, endpoint, command)
for i, response := range responses {
w.Write([]byte("<h2>" + html.EscapeString(servers[i]) + ": " + html.EscapeString(command) + "</h2>"))
if (endpoint == "bird" || endpoint == "bird6") && command == "show protocols" && len(response) > 4 && strings.ToLower(response[0:4]) == "name" {
var isIPv6 bool = endpoint[len(endpoint)-1] == '6'
summaryTable(w, isIPv6, response, servers[i])
} else {
smartWriter(w, response)
}
} }
templateFooter(w) return func(w http.ResponseWriter, r *http.Request) {
split := strings.Split(r.URL.Path[1:], "/")
urlCommands := strings.Join(split[3:], "/")
var backendCommand string
if strings.Contains(backendCommandPrimitive, "%") {
backendCommand = fmt.Sprintf(backendCommandPrimitive, urlCommands)
} else {
backendCommand = backendCommandPrimitive
}
templateHeader(w, r, "Bird-lg Go - "+html.EscapeString(endpoint+" "+backendCommand))
var servers []string = strings.Split(split[2], "+")
var responses []string = batchRequest(servers, endpoint, backendCommand)
for i, response := range responses {
w.Write([]byte("<h2>" + html.EscapeString(servers[i]) + ": " + html.EscapeString(backendCommand) + "</h2>"))
if (endpoint == "bird" || endpoint == "bird6") && backendCommand == "show protocols" && len(response) > 4 && strings.ToLower(response[0:4]) == "name" {
var isIPv6 bool = endpoint[len(endpoint)-1] == '6'
summaryTable(w, isIPv6, response, servers[i])
} else {
smartWriter(w, response)
}
}
templateFooter(w)
}
} }
func webHandlerBGPMap(w http.ResponseWriter, r *http.Request, endpoint string, command string) { func webHandlerBGPMap(endpoint string, command string) func(w http.ResponseWriter, r *http.Request) {
split := strings.Split(r.URL.Path[1:], "/") backendCommandPrimitive, commandPresent := (map[string]string{
urlCommands := strings.Join(split[3:], "/") "route_bgpmap": "show route for %s all",
"route_where_bgpmap": "show route where net ~ [ %s ] all",
command = (map[string]string{
"route_bgpmap": "show route for " + urlCommands + " all",
"route_where_bgpmap": "show route where net ~ [ " + urlCommands + " ] all",
})[command] })[command]
templateHeader(w, r, "Bird-lg Go - "+html.EscapeString(endpoint+" "+command)) if !commandPresent {
panic("invalid command: " + command)
}
var servers []string = strings.Split(split[2], "+") return func(w http.ResponseWriter, r *http.Request) {
split := strings.Split(r.URL.Path[1:], "/")
urlCommands := strings.Join(split[3:], "/")
var responses []string = batchRequest(servers, endpoint, command) var backendCommand string
w.Write([]byte(` if strings.Contains(backendCommandPrimitive, "%") {
<script> backendCommand = fmt.Sprintf(backendCommandPrimitive, urlCommands)
var viz = new Viz(); } else {
viz.renderSVGElement(` + "`" + birdRouteToGraphviz(servers, responses, urlCommands) + "`" + `) backendCommand = backendCommandPrimitive
.then(function(element) { }
document.body.appendChild(element);
})
.catch(error => {
document.body.appendChild("<pre>"+error+"</pre>")
});
</script>`))
templateFooter(w) templateHeader(w, r, "Bird-lg Go - "+html.EscapeString(endpoint+" "+backendCommand))
var servers []string = strings.Split(split[2], "+")
var responses []string = batchRequest(servers, endpoint, backendCommand)
w.Write([]byte(`
<script>
var viz = new Viz();
viz.renderSVGElement(` + "`" + birdRouteToGraphviz(servers, responses, urlCommands) + "`" + `)
.then(function(element) {
document.body.appendChild(element);
})
.catch(error => {
document.body.appendChild("<pre>"+error+"</pre>")
});
</script>`))
templateFooter(w)
}
} }
func webHandlerNavbarFormRedirect(w http.ResponseWriter, r *http.Request) { func webHandlerNavbarFormRedirect(w http.ResponseWriter, r *http.Request) {
@ -94,26 +121,24 @@ func webServerStart() {
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) { http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
http.Redirect(w, r, "/ipv4/summary/"+strings.Join(settingServers[:], "+"), 302) http.Redirect(w, r, "/ipv4/summary/"+strings.Join(settingServers[:], "+"), 302)
}) })
http.HandleFunc("/ipv4/summary/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "summary") }) http.HandleFunc("/ipv4/summary/", webBackendCommunicator("bird", "summary"))
http.HandleFunc("/ipv6/summary/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "summary") }) http.HandleFunc("/ipv6/summary/", webBackendCommunicator("bird6", "summary"))
http.HandleFunc("/ipv4/detail/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "detail") }) http.HandleFunc("/ipv4/detail/", webBackendCommunicator("bird", "detail"))
http.HandleFunc("/ipv6/detail/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "detail") }) http.HandleFunc("/ipv6/detail/", webBackendCommunicator("bird6", "detail"))
http.HandleFunc("/ipv4/route/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "route") }) http.HandleFunc("/ipv4/route/", webBackendCommunicator("bird", "route"))
http.HandleFunc("/ipv6/route/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "route") }) http.HandleFunc("/ipv6/route/", webBackendCommunicator("bird6", "route"))
http.HandleFunc("/ipv4/route_all/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "route_all") }) http.HandleFunc("/ipv4/route_all/", webBackendCommunicator("bird", "route_all"))
http.HandleFunc("/ipv6/route_all/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "route_all") }) http.HandleFunc("/ipv6/route_all/", webBackendCommunicator("bird6", "route_all"))
http.HandleFunc("/ipv4/route_bgpmap/", func(w http.ResponseWriter, r *http.Request) { webHandlerBGPMap(w, r, "bird", "route_bgpmap") }) http.HandleFunc("/ipv4/route_bgpmap/", webHandlerBGPMap("bird", "route_bgpmap"))
http.HandleFunc("/ipv6/route_bgpmap/", func(w http.ResponseWriter, r *http.Request) { webHandlerBGPMap(w, r, "bird6", "route_bgpmap") }) http.HandleFunc("/ipv6/route_bgpmap/", webHandlerBGPMap("bird6", "route_bgpmap"))
http.HandleFunc("/ipv4/route_where/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "route_where") }) http.HandleFunc("/ipv4/route_where/", webBackendCommunicator("bird", "route_where"))
http.HandleFunc("/ipv6/route_where/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "route_where") }) http.HandleFunc("/ipv6/route_where/", webBackendCommunicator("bird6", "route_where"))
http.HandleFunc("/ipv4/route_where_all/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "route_where_all") }) http.HandleFunc("/ipv4/route_where_all/", webBackendCommunicator("bird", "route_where_all"))
http.HandleFunc("/ipv6/route_where_all/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "route_where_all") }) http.HandleFunc("/ipv6/route_where_all/", webBackendCommunicator("bird6", "route_where_all"))
http.HandleFunc("/ipv4/route_where_bgpmap/", func(w http.ResponseWriter, r *http.Request) { webHandlerBGPMap(w, r, "bird", "route_where_bgpmap") }) http.HandleFunc("/ipv4/route_where_bgpmap/", webHandlerBGPMap("bird", "route_where_bgpmap"))
http.HandleFunc("/ipv6/route_where_bgpmap/", func(w http.ResponseWriter, r *http.Request) { webHandlerBGPMap(w, r, "bird6", "route_where_bgpmap") }) http.HandleFunc("/ipv6/route_where_bgpmap/", webHandlerBGPMap("bird6", "route_where_bgpmap"))
http.HandleFunc("/ipv4/traceroute/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "traceroute", "traceroute") }) http.HandleFunc("/ipv4/traceroute/", webBackendCommunicator("traceroute", "traceroute"))
http.HandleFunc("/ipv6/traceroute/", func(w http.ResponseWriter, r *http.Request) { http.HandleFunc("/ipv6/traceroute/", webBackendCommunicator("traceroute6", "traceroute"))
webBackendCommunicator(w, r, "traceroute6", "traceroute")
})
http.HandleFunc("/whois/", webHandlerWhois) http.HandleFunc("/whois/", webHandlerWhois)
http.HandleFunc("/redir/", webHandlerNavbarFormRedirect) http.HandleFunc("/redir/", webHandlerNavbarFormRedirect)
http.ListenAndServe(settingListen, nil) http.ListenAndServe(settingListen, nil)