package main import ( "net/http" "regexp" "sort" "strings" ) 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)", "route_generic": "show route ...", "generic": "show ...", "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 += "
" for _, line := range strings.Split(s, "\n") { var lineFormatted string if strings.HasPrefix(strings.TrimSpace(line), "BGP.as_path:") || strings.HasPrefix(strings.TrimSpace(line), "Neighbor AS:") || strings.HasPrefix(strings.TrimSpace(line), "Local AS:") { lineFormatted = regexp.MustCompile(`(\d+)`).ReplaceAllString(line, `${1}`) } else { lineFormatted = regexp.MustCompile(`([a-zA-Z0-9\-]*\.([a-zA-Z]{2,3}){1,2})(\s|$)`).ReplaceAllString(line, `${1}${3}`) lineFormatted = regexp.MustCompile(`\[AS(\d+)`).ReplaceAllString(lineFormatted, `[AS${1}`) lineFormatted = regexp.MustCompile(`(\d+\.\d+\.\d+\.\d+)`).ReplaceAllString(lineFormatted, `${1}`) lineFormatted = regexp.MustCompile(`(?i)(([a-f\d]{0,4}:){3,10}[a-f\d]{0,4})`).ReplaceAllString(lineFormatted, `${1}`) } result += lineFormatted + "\n" } result += "" return result } type summaryTableArguments struct { Headers []string Lines [][]string } // 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(strings.TrimSpace(data), "\n") if len(stringsSplitted) <= 1 { // Likely backend returned an error message result = "
" + strings.TrimSpace(data) + "" } else { // Draw the table head result += "
" + colTrimmed + " | " } result += "" stringsWithoutTitle := stringsSplitted[1:] sort.Strings(stringsWithoutTitle) for _, line := range stringsWithoutTitle { // Ignore empty lines line = strings.TrimSpace(line) if len(line) == 0 { continue } // Parse a total of 6 columns from bird summary lineSplitted := regexp.MustCompile(`(\w+)(\s+)(\w+)(\s+)([\w-]+)(\s+)(\w+)(\s+)([0-9\- :]+)(.*)`).FindStringSubmatch(line) var row = [6]string{ strings.TrimSpace(lineSplitted[1]), strings.TrimSpace(lineSplitted[3]), strings.TrimSpace(lineSplitted[5]), strings.TrimSpace(lineSplitted[7]), strings.TrimSpace(lineSplitted[9]), strings.TrimSpace(lineSplitted[10]), } // Draw the row in red if the link isn't up result += "||
---|---|---|
" + row[0] + " | " } else { result += "" + row[0] + " | " } // Draw the other cells for i := 1; i < 6; i++ { result += "" + row[i] + " | " } result += "