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 } // 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 = `
")) 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 += "" + word + "" } 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 += "" + strings.Split(word, "%")[0] + "" 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 += "" + strings.Split(word, "/")[0] + "" 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 += "" + word + "" } else { // Just an ordinary word, print it and done lineFormatted += word } } lineFormatted += "\n" w.Write([]byte(lineFormatted)) } w.Write([]byte("")) } // 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("
" + data + "")) w.Write([]byte("
" + row[i] + " | ")) } w.Write([]byte("")) } else { // Draw the row in red if the link isn't up w.Write([]byte("||
---|---|---|
" + row[0] + " | ")) } else { w.Write([]byte("" + row[0] + " | ")) } // Draw the other cells for i := 1; i < 6; i++ { w.Write([]byte("" + row[i] + " | ")) } w.Write([]byte("