frontend: use template for rendering (partial)
This commit is contained in:
parent
4df9006c81
commit
3fe3def49f
@ -1,128 +1,40 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net"
|
"text/template"
|
||||||
"net/http"
|
|
||||||
"sort"
|
|
||||||
"strconv"
|
|
||||||
"strings"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Helper to check if the IP is valid
|
type tmplArguments struct {
|
||||||
func isIP(s string) bool {
|
// Global options
|
||||||
return nil != net.ParseIP(s)
|
Options map[string]string
|
||||||
|
Servers []string
|
||||||
|
|
||||||
|
// Parameters related to current request
|
||||||
|
AllServersLinkActive bool
|
||||||
|
AllServersURL string
|
||||||
|
// Whois specific handling (for its unique URL)
|
||||||
|
IsWhois bool
|
||||||
|
WhoisTarget string
|
||||||
|
|
||||||
|
URLProto string
|
||||||
|
URLOption string
|
||||||
|
URLServer string
|
||||||
|
URLCommand string
|
||||||
|
|
||||||
|
// Generated content to be displayed
|
||||||
|
Title string
|
||||||
|
Content string
|
||||||
}
|
}
|
||||||
|
|
||||||
// Helper to check if the number is valid
|
var tmpl = template.Must(template.New("tmpl").Parse(`
|
||||||
func isNumber(s string) bool {
|
|
||||||
_, err := strconv.Atoi(s)
|
|
||||||
return nil == err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Print HTML header to the given http response
|
|
||||||
func templateHeader(w http.ResponseWriter, r *http.Request, title string) {
|
|
||||||
path := r.URL.Path[1:]
|
|
||||||
split := strings.Split(path, "/")
|
|
||||||
|
|
||||||
// Mark if the URL is for a whois query
|
|
||||||
var isWhois bool = split[0] == "whois"
|
|
||||||
var whoisTarget string = strings.Join(split[1:], "/")
|
|
||||||
|
|
||||||
// Use a default URL if the request URL is too short
|
|
||||||
// The URL is for return to IPv4 summary page
|
|
||||||
if len(split) < 3 {
|
|
||||||
path = "ipv4/summary/" + strings.Join(setting.servers, "+") + "/"
|
|
||||||
} else if len(split) == 3 {
|
|
||||||
path += "/"
|
|
||||||
}
|
|
||||||
|
|
||||||
split = strings.Split(path, "/")
|
|
||||||
|
|
||||||
// Compose URLs for link in navbar
|
|
||||||
ipv4URL := "/" + strings.Join([]string{"ipv4", split[1], split[2], strings.Join(split[3:], "/")}, "/")
|
|
||||||
ipv6URL := "/" + strings.Join([]string{"ipv6", split[1], split[2], strings.Join(split[3:], "/")}, "/")
|
|
||||||
allURL := "/" + strings.Join([]string{split[0], split[1], strings.Join(setting.servers, "+"), strings.Join(split[3:], "/")}, "/")
|
|
||||||
|
|
||||||
// Check if the "All Server" link should be marked as active
|
|
||||||
var serverAllActive bool = strings.ToLower(split[2]) == strings.ToLower(strings.Join(setting.servers, "+"))
|
|
||||||
|
|
||||||
// Print the IPv4, IPv6, All Servers link in navbar
|
|
||||||
var serverNavigation string = `
|
|
||||||
<li class="nav-item"><a class="nav-link` + (map[bool]string{true: " active"})[strings.ToLower(split[0]) == "ipv4"] + `" href="` + ipv4URL + `"> IPv4 </a></li>
|
|
||||||
<li class="nav-item"><a class="nav-link` + (map[bool]string{true: " active"})[strings.ToLower(split[0]) == "ipv6"] + `" href="` + ipv6URL + `"> IPv6 </a></li>
|
|
||||||
<span class="navbar-text">|</span>
|
|
||||||
<li class="nav-item">
|
|
||||||
<a class="nav-link` + (map[bool]string{true: " active"})[serverAllActive] + `" href="` + allURL + `"> All Servers </a>
|
|
||||||
</li>`
|
|
||||||
|
|
||||||
// Add a link for each of the servers
|
|
||||||
for _, server := range setting.servers {
|
|
||||||
var serverActive string
|
|
||||||
if split[2] == server {
|
|
||||||
serverActive = " active"
|
|
||||||
}
|
|
||||||
serverURL := "/" + strings.Join([]string{split[0], split[1], server, strings.Join(split[3:], "/")}, "/")
|
|
||||||
|
|
||||||
serverNavigation += `
|
|
||||||
<li class="nav-item">
|
|
||||||
<a class="nav-link` + serverActive + `" href="` + serverURL + `">` + server + `</a>
|
|
||||||
</li>`
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add the options in navbar form, and check if they are active
|
|
||||||
var optionKeys = []string{
|
|
||||||
"summary",
|
|
||||||
"detail",
|
|
||||||
"route",
|
|
||||||
"route_all",
|
|
||||||
"route_bgpmap",
|
|
||||||
"route_where",
|
|
||||||
"route_where_all",
|
|
||||||
"route_where_bgpmap",
|
|
||||||
"whois",
|
|
||||||
"traceroute",
|
|
||||||
}
|
|
||||||
var optionDisplays = []string{
|
|
||||||
"show protocol",
|
|
||||||
"show protocol all",
|
|
||||||
"show route for ...",
|
|
||||||
"show route for ... all",
|
|
||||||
"show route for ... (bgpmap)",
|
|
||||||
"show route where net ~ [ ... ]",
|
|
||||||
"show route where net ~ [ ... ] all",
|
|
||||||
"show route where net ~ [ ... ] (bgpmap)",
|
|
||||||
"whois ...",
|
|
||||||
"traceroute ...",
|
|
||||||
}
|
|
||||||
|
|
||||||
var options string
|
|
||||||
for optionKeyID, optionKey := range optionKeys {
|
|
||||||
options += "<option value=\"" + optionKey + "\""
|
|
||||||
if (optionKey == "whois" && isWhois) || optionKey == split[1] {
|
|
||||||
options += " selected"
|
|
||||||
}
|
|
||||||
options += ">" + optionDisplays[optionKeyID] + "</option>"
|
|
||||||
}
|
|
||||||
|
|
||||||
var target string
|
|
||||||
if isWhois {
|
|
||||||
// This is a whois request, use original path URL instead of the modified one
|
|
||||||
// and extract the target
|
|
||||||
target = whoisTarget
|
|
||||||
} else if len(split) >= 4 {
|
|
||||||
// This is a normal request, just extract the target
|
|
||||||
target = strings.Join(split[3:], "/")
|
|
||||||
}
|
|
||||||
|
|
||||||
w.Write([]byte(`
|
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="zh-CN" lang="zh-CN" class="no-js">
|
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en-US" lang="en-US" class="no-js">
|
||||||
<head>
|
<head>
|
||||||
<meta http-equiv="Content-Type" content="text/html;charset=UTF-8" />
|
<meta http-equiv="Content-Type" content="text/html;charset=UTF-8" />
|
||||||
<meta http-equiv="X-UA-Compatible" content="IE=edge"/>
|
<meta http-equiv="X-UA-Compatible" content="IE=edge"/>
|
||||||
<meta name="viewport" content="width=device-width,initial-scale=1,shrink-to-fit=no"/>
|
<meta name="viewport" content="width=device-width,initial-scale=1,shrink-to-fit=no"/>
|
||||||
<meta name="renderer" content="webkit"/>
|
<meta name="renderer" content="webkit"/>
|
||||||
<title>` + title + `</title>
|
<title>{{ .Title }}</title>
|
||||||
<link href="https://cdn.jsdelivr.net/npm/bootstrap@4.4.1/dist/css/bootstrap.min.css" rel="stylesheet">
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap@4.4.1/dist/css/bootstrap.min.css" rel="stylesheet">
|
||||||
<script src="https://cdn.jsdelivr.net/npm/viz.js@2.1.2/viz.min.js" crossorigin="anonymous"></script>
|
<script src="https://cdn.jsdelivr.net/npm/viz.js@2.1.2/viz.min.js" crossorigin="anonymous"></script>
|
||||||
<script src="https://cdn.jsdelivr.net/npm/viz.js@2.1.2/lite.render.js" crossorigin="anonymous"></script>
|
<script src="https://cdn.jsdelivr.net/npm/viz.js@2.1.2/lite.render.js" crossorigin="anonymous"></script>
|
||||||
@ -136,13 +48,35 @@ func templateHeader(w http.ResponseWriter, r *http.Request, title string) {
|
|||||||
</button>
|
</button>
|
||||||
|
|
||||||
<div class="collapse navbar-collapse" id="navbarSupportedContent">
|
<div class="collapse navbar-collapse" id="navbarSupportedContent">
|
||||||
<ul class="navbar-nav mr-auto">` + serverNavigation + `</ul>
|
<ul class="navbar-nav mr-auto">
|
||||||
|
<li class="nav-item"><a class="nav-link{{ if eq "ipv4" .URLProto }} active{{ end }}" href="/ipv4/{{ .URLOption }}/{{ .URLServer }}/{{ .URLCommand }}"> IPv4 </a></li>
|
||||||
|
<li class="nav-item"><a class="nav-link{{ if eq "ipv6" .URLProto }} active{{ end }}" href="/ipv6/{{ .URLOption }}/{{ .URLServer }}/{{ .URLCommand }}"> IPv6 </a></li>
|
||||||
|
<span class="navbar-text">|</span>
|
||||||
|
<li class="nav-item">
|
||||||
|
<a class="nav-link{{ if .AllServersLinkActive }} active{{ end }}" href="/{{ .URLProto }}/{{ .URLOption }}/{{ .AllServersURL }}/{{ .URLCommand }}"> All Servers </a>
|
||||||
|
</li>
|
||||||
|
{{ range $k, $v := .Servers }}
|
||||||
|
<li class="nav-item">
|
||||||
|
<a class="nav-link{{ if eq $.URLServer $v }} active{{ end }}" href="/{{ $.URLProto }}/{{ $.URLOption }}/{{ $v }}/{{ $.URLCommand }}">{{ $v }}</a>
|
||||||
|
</li>
|
||||||
|
{{ end }}
|
||||||
|
</ul>
|
||||||
|
{{ $option := .URLOption }}
|
||||||
|
{{ $target := .URLCommand }}
|
||||||
|
{{ if .IsWhois }}
|
||||||
|
{{ $option = "whois" }}
|
||||||
|
{{ $target = .WhoisTarget }}
|
||||||
|
{{ end }}
|
||||||
<form class="form-inline" action="/redir" method="GET">
|
<form class="form-inline" action="/redir" method="GET">
|
||||||
<div class="input-group">
|
<div class="input-group">
|
||||||
<select name="action" class="form-control">` + options + `</select>
|
<select name="action" class="form-control">
|
||||||
<input name="proto" class="d-none" value="` + split[0] + `">
|
{{ range $k, $v := .Options }}
|
||||||
<input name="server" class="d-none" value="` + split[2] + `">
|
<option value="{{ $k }}"{{ if eq $k $option }} selected{{end}}>{{ $v }}</option>
|
||||||
<input name="target" class="form-control" placeholder="Target" aria-label="Target" value="` + target + `">
|
{{ end }}
|
||||||
|
</select>
|
||||||
|
<input name="proto" class="d-none" value="{{ .URLProto }}">
|
||||||
|
<input name="server" class="d-none" value="{{ .URLServer }}">
|
||||||
|
<input name="target" class="form-control" placeholder="Target" aria-label="Target" value="{{ $target }}">
|
||||||
<div class="input-group-append">
|
<div class="input-group-append">
|
||||||
<button class="btn btn-outline-success" type="submit">»</button>
|
<button class="btn btn-outline-success" type="submit">»</button>
|
||||||
</div>
|
</div>
|
||||||
@ -152,138 +86,8 @@ func templateHeader(w http.ResponseWriter, r *http.Request, title string) {
|
|||||||
</nav>
|
</nav>
|
||||||
|
|
||||||
<div class="container">
|
<div class="container">
|
||||||
`))
|
{{ .Content }}
|
||||||
}
|
|
||||||
|
|
||||||
// Print HTML footer to http response
|
|
||||||
func templateFooter(w http.ResponseWriter) {
|
|
||||||
w.Write([]byte(`
|
|
||||||
</div>
|
</div>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
`))
|
`))
|
||||||
}
|
|
||||||
|
|
||||||
// Write the given text to http response, and add whois links for
|
|
||||||
// ASNs and IP addresses
|
|
||||||
func smartWriter(w http.ResponseWriter, s string) {
|
|
||||||
w.Write([]byte("<pre>"))
|
|
||||||
for _, line := range strings.Split(s, "\n") {
|
|
||||||
var isASes bool = false
|
|
||||||
|
|
||||||
var lineFormatted string
|
|
||||||
words := strings.Split(line, " ")
|
|
||||||
|
|
||||||
for wordID, word := range words {
|
|
||||||
if len(word) == 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if wordID > 0 && (len(words[wordID-1]) == 0 || words[wordID-1][len(words[wordID-1])-1] == ':') {
|
|
||||||
// Insert TAB if there are multiple spaces before this word
|
|
||||||
lineFormatted += "\t"
|
|
||||||
} else {
|
|
||||||
lineFormatted += " "
|
|
||||||
}
|
|
||||||
|
|
||||||
if isIP(word) {
|
|
||||||
// Add whois link to the IP, handles IPv4 and IPv6
|
|
||||||
lineFormatted += "<a href=\"/whois/" + word + "\">" + word + "</a>"
|
|
||||||
} else if len(strings.Split(word, "%")) == 2 && isIP(strings.Split(word, "%")[0]) {
|
|
||||||
// IPv6 link-local with interface name, like fd00::1%eth0
|
|
||||||
// Add whois link to address part
|
|
||||||
lineFormatted += "<a href=\"/whois/" + strings.Split(word, "%")[0] + "\">" + strings.Split(word, "%")[0] + "</a>"
|
|
||||||
lineFormatted += "%" + strings.Split(word, "%")[1]
|
|
||||||
} else if len(strings.Split(word, "/")) == 2 && isIP(strings.Split(word, "/")[0]) {
|
|
||||||
// IP with a CIDR range, like 192.168.0.1/24
|
|
||||||
// Add whois link to first part
|
|
||||||
lineFormatted += "<a href=\"/whois/" + strings.Split(word, "/")[0] + "\">" + strings.Split(word, "/")[0] + "</a>"
|
|
||||||
lineFormatted += "/" + strings.Split(word, "/")[1]
|
|
||||||
} else if word == "AS:" || word == "\tBGP.as_path:" {
|
|
||||||
// Bird will output ASNs later
|
|
||||||
isASes = true
|
|
||||||
lineFormatted += word
|
|
||||||
} else if isASes && isNumber(strings.Trim(word, "()")) {
|
|
||||||
// Remove brackets in path caused by confederation
|
|
||||||
wordNum := strings.Trim(word, "()")
|
|
||||||
// Bird is outputing ASNs, add whois for them
|
|
||||||
lineFormatted += "<a href=\"/whois/AS" + wordNum + "\">" + word + "</a>"
|
|
||||||
} else {
|
|
||||||
// Just an ordinary word, print it and done
|
|
||||||
lineFormatted += word
|
|
||||||
}
|
|
||||||
}
|
|
||||||
lineFormatted += "\n"
|
|
||||||
w.Write([]byte(lineFormatted))
|
|
||||||
}
|
|
||||||
w.Write([]byte("</pre>"))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output a table for the summary page
|
|
||||||
func summaryTable(w http.ResponseWriter, isIPv6 bool, data string, serverName string) {
|
|
||||||
// Sort the table, excluding title row
|
|
||||||
stringsSplitted := strings.Split(data, "\n")
|
|
||||||
if len(stringsSplitted) > 1 {
|
|
||||||
stringsWithoutTitle := stringsSplitted[1:]
|
|
||||||
sort.Strings(stringsWithoutTitle)
|
|
||||||
data = stringsSplitted[0] + "\n" + strings.Join(stringsWithoutTitle, "\n")
|
|
||||||
}
|
|
||||||
|
|
||||||
// w.Write([]byte("<pre>" + data + "</pre>"))
|
|
||||||
w.Write([]byte("<table class=\"table table-striped table-bordered table-sm\">"))
|
|
||||||
for lineID, line := range strings.Split(data, "\n") {
|
|
||||||
var row [6]string
|
|
||||||
var rowIndex int = 0
|
|
||||||
|
|
||||||
words := strings.Split(line, " ")
|
|
||||||
for wordID, word := range words {
|
|
||||||
if len(word) == 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if rowIndex < 4 {
|
|
||||||
row[rowIndex] += word
|
|
||||||
rowIndex++
|
|
||||||
} else if len(words[wordID-1]) == 0 && rowIndex < len(row)-1 {
|
|
||||||
if len(row[rowIndex]) > 0 {
|
|
||||||
rowIndex++
|
|
||||||
}
|
|
||||||
row[rowIndex] += word
|
|
||||||
} else {
|
|
||||||
row[rowIndex] += " " + word
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ignore empty lines
|
|
||||||
if len(row[0]) == 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if lineID == 0 {
|
|
||||||
// Draw the table head
|
|
||||||
w.Write([]byte("<thead>"))
|
|
||||||
for i := 0; i < 6; i++ {
|
|
||||||
w.Write([]byte("<th scope=\"col\">" + row[i] + "</th>"))
|
|
||||||
}
|
|
||||||
w.Write([]byte("</thead><tbody>"))
|
|
||||||
} else {
|
|
||||||
// Draw the row in red if the link isn't up
|
|
||||||
w.Write([]byte("<tr class=\"" + (map[string]string{
|
|
||||||
"up": "table-success",
|
|
||||||
"down": "table-secondary",
|
|
||||||
"start": "table-danger",
|
|
||||||
"passive": "table-info",
|
|
||||||
})[row[3]] + "\">"))
|
|
||||||
// Add link to detail for first column
|
|
||||||
if isIPv6 {
|
|
||||||
w.Write([]byte("<td><a href=\"/ipv6/detail/" + serverName + "/" + row[0] + "\">" + row[0] + "</a></td>"))
|
|
||||||
} else {
|
|
||||||
w.Write([]byte("<td><a href=\"/ipv4/detail/" + serverName + "/" + row[0] + "\">" + row[0] + "</a></td>"))
|
|
||||||
}
|
|
||||||
// Draw the other cells
|
|
||||||
for i := 1; i < 6; i++ {
|
|
||||||
w.Write([]byte("<td>" + row[i] + "</td>"))
|
|
||||||
}
|
|
||||||
w.Write([]byte("</tr>"))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
w.Write([]byte("</tbody></table>"))
|
|
||||||
}
|
|
||||||
|
198
frontend/templateHelper.go
Normal file
198
frontend/templateHelper.go
Normal file
@ -0,0 +1,198 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net"
|
||||||
|
"net/http"
|
||||||
|
"sort"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Helper to check if the IP is valid
|
||||||
|
func isIP(s string) bool {
|
||||||
|
return nil != net.ParseIP(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper to check if the number is valid
|
||||||
|
func isNumber(s string) bool {
|
||||||
|
_, err := strconv.Atoi(s)
|
||||||
|
return nil == err
|
||||||
|
}
|
||||||
|
|
||||||
|
func renderTemplate(w http.ResponseWriter, r *http.Request, title string, content string) {
|
||||||
|
path := r.URL.Path[1:]
|
||||||
|
split := strings.SplitN(path, "/", 4)
|
||||||
|
|
||||||
|
isWhois := strings.ToLower(split[0]) == "whois"
|
||||||
|
whoisTarget := strings.Join(split[1:], "/")
|
||||||
|
|
||||||
|
// Use a default URL if the request URL is too short
|
||||||
|
// The URL is for return to IPv4 summary page
|
||||||
|
if len(split) < 3 {
|
||||||
|
path = "ipv4/summary/" + strings.Join(setting.servers, "+") + "/"
|
||||||
|
} else if len(split) == 3 {
|
||||||
|
path += "/"
|
||||||
|
}
|
||||||
|
|
||||||
|
split = strings.SplitN(path, "/", 4)
|
||||||
|
|
||||||
|
var args tmplArguments
|
||||||
|
args.Options = map[string]string{
|
||||||
|
"summary": "show protocol",
|
||||||
|
"detail": "show protocol all",
|
||||||
|
"route": "show route for ...",
|
||||||
|
"route_all": "show route for ... all",
|
||||||
|
"route_bgpmap": "show route for ... (bgpmap)",
|
||||||
|
"route_where": "show route where net ~ [ ... ]",
|
||||||
|
"route_where_all": "show route where net ~ [ ... ] all",
|
||||||
|
"route_where_bgpmap": "show route where net ~ [ ... ] (bgpmap)",
|
||||||
|
"whois": "whois ...",
|
||||||
|
"traceroute": "traceroute ...",
|
||||||
|
}
|
||||||
|
args.Servers = setting.servers
|
||||||
|
args.AllServersLinkActive = strings.ToLower(split[2]) == strings.ToLower(strings.Join(setting.servers, "+"))
|
||||||
|
args.AllServersURL = strings.Join(setting.servers, "+")
|
||||||
|
args.IsWhois = isWhois
|
||||||
|
args.WhoisTarget = whoisTarget
|
||||||
|
|
||||||
|
args.URLProto = strings.ToLower(split[0])
|
||||||
|
args.URLOption = strings.ToLower(split[1])
|
||||||
|
args.URLServer = strings.ToLower(split[2])
|
||||||
|
args.URLCommand = split[3]
|
||||||
|
|
||||||
|
args.Title = title
|
||||||
|
args.Content = content
|
||||||
|
|
||||||
|
err := tmpl.Execute(w, args)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write the given text to http response, and add whois links for
|
||||||
|
// ASNs and IP addresses
|
||||||
|
func smartFormatter(s string) string {
|
||||||
|
var result string
|
||||||
|
result += "<pre>"
|
||||||
|
for _, line := range strings.Split(s, "\n") {
|
||||||
|
var isASes bool = false
|
||||||
|
|
||||||
|
var lineFormatted string
|
||||||
|
words := strings.Split(line, " ")
|
||||||
|
|
||||||
|
for wordID, word := range words {
|
||||||
|
if len(word) == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if wordID > 0 && (len(words[wordID-1]) == 0 || words[wordID-1][len(words[wordID-1])-1] == ':') {
|
||||||
|
// Insert TAB if there are multiple spaces before this word
|
||||||
|
lineFormatted += "\t"
|
||||||
|
} else {
|
||||||
|
lineFormatted += " "
|
||||||
|
}
|
||||||
|
|
||||||
|
if isIP(word) {
|
||||||
|
// Add whois link to the IP, handles IPv4 and IPv6
|
||||||
|
lineFormatted += "<a href=\"/whois/" + word + "\">" + word + "</a>"
|
||||||
|
} else if len(strings.Split(word, "%")) == 2 && isIP(strings.Split(word, "%")[0]) {
|
||||||
|
// IPv6 link-local with interface name, like fd00::1%eth0
|
||||||
|
// Add whois link to address part
|
||||||
|
lineFormatted += "<a href=\"/whois/" + strings.Split(word, "%")[0] + "\">" + strings.Split(word, "%")[0] + "</a>"
|
||||||
|
lineFormatted += "%" + strings.Split(word, "%")[1]
|
||||||
|
} else if len(strings.Split(word, "/")) == 2 && isIP(strings.Split(word, "/")[0]) {
|
||||||
|
// IP with a CIDR range, like 192.168.0.1/24
|
||||||
|
// Add whois link to first part
|
||||||
|
lineFormatted += "<a href=\"/whois/" + strings.Split(word, "/")[0] + "\">" + strings.Split(word, "/")[0] + "</a>"
|
||||||
|
lineFormatted += "/" + strings.Split(word, "/")[1]
|
||||||
|
} else if word == "AS:" || word == "\tBGP.as_path:" {
|
||||||
|
// Bird will output ASNs later
|
||||||
|
isASes = true
|
||||||
|
lineFormatted += word
|
||||||
|
} else if isASes && isNumber(strings.Trim(word, "()")) {
|
||||||
|
// Remove brackets in path caused by confederation
|
||||||
|
wordNum := strings.Trim(word, "()")
|
||||||
|
// Bird is outputing ASNs, add whois for them
|
||||||
|
lineFormatted += "<a href=\"/whois/AS" + wordNum + "\">" + word + "</a>"
|
||||||
|
} else {
|
||||||
|
// Just an ordinary word, print it and done
|
||||||
|
lineFormatted += word
|
||||||
|
}
|
||||||
|
}
|
||||||
|
lineFormatted += "\n"
|
||||||
|
result += lineFormatted
|
||||||
|
}
|
||||||
|
result += "</pre>"
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output a table for the summary page
|
||||||
|
func summaryTable(isIPv6 bool, data string, serverName string) string {
|
||||||
|
var result string
|
||||||
|
|
||||||
|
// Sort the table, excluding title row
|
||||||
|
stringsSplitted := strings.Split(data, "\n")
|
||||||
|
if len(stringsSplitted) > 1 {
|
||||||
|
stringsWithoutTitle := stringsSplitted[1:]
|
||||||
|
sort.Strings(stringsWithoutTitle)
|
||||||
|
data = stringsSplitted[0] + "\n" + strings.Join(stringsWithoutTitle, "\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
result += "<table class=\"table table-striped table-bordered table-sm\">"
|
||||||
|
for lineID, line := range strings.Split(data, "\n") {
|
||||||
|
var row [6]string
|
||||||
|
var rowIndex int = 0
|
||||||
|
|
||||||
|
words := strings.Split(line, " ")
|
||||||
|
for wordID, word := range words {
|
||||||
|
if len(word) == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if rowIndex < 4 {
|
||||||
|
row[rowIndex] += word
|
||||||
|
rowIndex++
|
||||||
|
} else if len(words[wordID-1]) == 0 && rowIndex < len(row)-1 {
|
||||||
|
if len(row[rowIndex]) > 0 {
|
||||||
|
rowIndex++
|
||||||
|
}
|
||||||
|
row[rowIndex] += word
|
||||||
|
} else {
|
||||||
|
row[rowIndex] += " " + word
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ignore empty lines
|
||||||
|
if len(row[0]) == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if lineID == 0 {
|
||||||
|
// Draw the table head
|
||||||
|
result += "<thead>"
|
||||||
|
for i := 0; i < 6; i++ {
|
||||||
|
result += "<th scope=\"col\">" + row[i] + "</th>"
|
||||||
|
}
|
||||||
|
result += "</thead><tbody>"
|
||||||
|
} else {
|
||||||
|
// Draw the row in red if the link isn't up
|
||||||
|
result += "<tr class=\"" + (map[string]string{
|
||||||
|
"up": "table-success",
|
||||||
|
"down": "table-secondary",
|
||||||
|
"start": "table-danger",
|
||||||
|
"passive": "table-info",
|
||||||
|
})[row[3]] + "\">"
|
||||||
|
// Add link to detail for first column
|
||||||
|
if isIPv6 {
|
||||||
|
result += "<td><a href=\"/ipv6/detail/" + serverName + "/" + row[0] + "\">" + row[0] + "</a></td>"
|
||||||
|
} else {
|
||||||
|
result += "<td><a href=\"/ipv4/detail/" + serverName + "/" + row[0] + "\">" + row[0] + "</a></td>"
|
||||||
|
}
|
||||||
|
// Draw the other cells
|
||||||
|
for i := 1; i < 6; i++ {
|
||||||
|
result += "<td>" + row[i] + "</td>"
|
||||||
|
}
|
||||||
|
result += "</tr>"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
result += "</tbody></table>"
|
||||||
|
return result
|
||||||
|
}
|
@ -10,12 +10,11 @@ import (
|
|||||||
func webHandlerWhois(w http.ResponseWriter, r *http.Request) {
|
func webHandlerWhois(w http.ResponseWriter, r *http.Request) {
|
||||||
var target string = r.URL.Path[len("/whois/"):]
|
var target string = r.URL.Path[len("/whois/"):]
|
||||||
|
|
||||||
templateHeader(w, r, "Bird-lg Go - whois "+html.EscapeString(target))
|
renderTemplate(
|
||||||
|
w, r,
|
||||||
w.Write([]byte("<h2>whois " + html.EscapeString(target) + "</h2>"))
|
"Bird-lg Go - whois "+html.EscapeString(target),
|
||||||
smartWriter(w, whois(target))
|
"<h2>whois "+html.EscapeString(target)+"</h2>"+smartFormatter(whois(target)),
|
||||||
|
)
|
||||||
templateFooter(w)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func webBackendCommunicator(endpoint string, command string) func(w http.ResponseWriter, r *http.Request) {
|
func webBackendCommunicator(endpoint string, command string) func(w http.ResponseWriter, r *http.Request) {
|
||||||
@ -34,8 +33,11 @@ func webBackendCommunicator(endpoint string, command string) func(w http.Respons
|
|||||||
}
|
}
|
||||||
|
|
||||||
return func(w http.ResponseWriter, r *http.Request) {
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
split := strings.Split(r.URL.Path[1:], "/")
|
split := strings.SplitN(r.URL.Path[1:], "/", 4)
|
||||||
urlCommands := strings.Join(split[3:], "/")
|
var urlCommands string
|
||||||
|
if len(split) >= 4 {
|
||||||
|
urlCommands = split[3]
|
||||||
|
}
|
||||||
|
|
||||||
var backendCommand string
|
var backendCommand string
|
||||||
if strings.Contains(backendCommandPrimitive, "%") {
|
if strings.Contains(backendCommandPrimitive, "%") {
|
||||||
@ -45,22 +47,24 @@ func webBackendCommunicator(endpoint string, command string) func(w http.Respons
|
|||||||
}
|
}
|
||||||
backendCommand = strings.TrimSpace(backendCommand)
|
backendCommand = strings.TrimSpace(backendCommand)
|
||||||
|
|
||||||
templateHeader(w, r, "Bird-lg Go - "+html.EscapeString(endpoint+" "+backendCommand))
|
|
||||||
|
|
||||||
var servers []string = strings.Split(split[2], "+")
|
var servers []string = strings.Split(split[2], "+")
|
||||||
|
|
||||||
var responses []string = batchRequest(servers, endpoint, backendCommand)
|
var responses []string = batchRequest(servers, endpoint, backendCommand)
|
||||||
|
var result string
|
||||||
for i, response := range responses {
|
for i, response := range responses {
|
||||||
w.Write([]byte("<h2>" + html.EscapeString(servers[i]) + ": " + html.EscapeString(backendCommand) + "</h2>"))
|
result += "<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" {
|
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'
|
var isIPv6 bool = endpoint[len(endpoint)-1] == '6'
|
||||||
summaryTable(w, isIPv6, response, servers[i])
|
result += summaryTable(isIPv6, response, servers[i])
|
||||||
} else {
|
} else {
|
||||||
smartWriter(w, response)
|
result += smartFormatter(response)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
templateFooter(w)
|
renderTemplate(
|
||||||
|
w, r,
|
||||||
|
"Bird-lg Go - "+html.EscapeString(endpoint+" "+backendCommand),
|
||||||
|
result,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -85,24 +89,22 @@ func webHandlerBGPMap(endpoint string, command string) func(w http.ResponseWrite
|
|||||||
backendCommand = backendCommandPrimitive
|
backendCommand = backendCommandPrimitive
|
||||||
}
|
}
|
||||||
|
|
||||||
templateHeader(w, r, "Bird-lg Go - "+html.EscapeString(endpoint+" "+backendCommand))
|
|
||||||
|
|
||||||
var servers []string = strings.Split(split[2], "+")
|
var servers []string = strings.Split(split[2], "+")
|
||||||
|
|
||||||
var responses []string = batchRequest(servers, endpoint, backendCommand)
|
var responses []string = batchRequest(servers, endpoint, backendCommand)
|
||||||
w.Write([]byte(`
|
renderTemplate(
|
||||||
<script>
|
w, r,
|
||||||
var viz = new Viz();
|
"Bird-lg Go - "+html.EscapeString(endpoint+" "+backendCommand),
|
||||||
viz.renderSVGElement(` + "`" + birdRouteToGraphviz(servers, responses, urlCommands) + "`" + `)
|
`<script>
|
||||||
.then(function(element) {
|
var viz = new Viz();
|
||||||
document.body.appendChild(element);
|
viz.renderSVGElement(`+"`"+birdRouteToGraphviz(servers, responses, urlCommands)+"`"+`)
|
||||||
})
|
.then(function(element) {
|
||||||
.catch(error => {
|
document.body.appendChild(element);
|
||||||
document.body.appendChild("<pre>"+error+"</pre>")
|
})
|
||||||
});
|
.catch(error => {
|
||||||
</script>`))
|
document.body.appendChild("<pre>"+error+"</pre>")
|
||||||
|
});
|
||||||
templateFooter(w)
|
</script>`,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -141,7 +143,7 @@ func webServerStart() {
|
|||||||
http.HandleFunc("/ipv4/traceroute/", webBackendCommunicator("traceroute", "traceroute"))
|
http.HandleFunc("/ipv4/traceroute/", webBackendCommunicator("traceroute", "traceroute"))
|
||||||
http.HandleFunc("/ipv6/traceroute/", webBackendCommunicator("traceroute6", "traceroute"))
|
http.HandleFunc("/ipv6/traceroute/", webBackendCommunicator("traceroute6", "traceroute"))
|
||||||
http.HandleFunc("/whois/", webHandlerWhois)
|
http.HandleFunc("/whois/", webHandlerWhois)
|
||||||
http.HandleFunc("/redir/", webHandlerNavbarFormRedirect)
|
http.HandleFunc("/redir", webHandlerNavbarFormRedirect)
|
||||||
http.HandleFunc("/telegram/", webHandlerTelegramBot)
|
http.HandleFunc("/telegram/", webHandlerTelegramBot)
|
||||||
http.ListenAndServe(setting.listen, nil)
|
http.ListenAndServe(setting.listen, nil)
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user