Multiple UI and functionality fixes
This commit is contained in:
parent
5d91c9d6d1
commit
19fd44c28e
@ -1,58 +1,61 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"io/ioutil"
|
||||||
"net/url"
|
"net/http"
|
||||||
"io/ioutil"
|
"net/url"
|
||||||
"strconv"
|
"strconv"
|
||||||
)
|
)
|
||||||
|
|
||||||
type channelData struct {
|
type channelData struct {
|
||||||
id int
|
id int
|
||||||
data string
|
data string
|
||||||
}
|
}
|
||||||
|
|
||||||
// Send commands to lgproxy instances in parallel, and retrieve their responses
|
// Send commands to lgproxy instances in parallel, and retrieve their responses
|
||||||
func batchRequest(servers []string, endpoint string, command string) []string {
|
func batchRequest(servers []string, endpoint string, command string) []string {
|
||||||
// Channel and array for storing responses
|
// Channel and array for storing responses
|
||||||
var ch chan channelData = make(chan channelData)
|
var ch chan channelData = make(chan channelData)
|
||||||
var response_array []string = make([]string, len(servers))
|
var responseArray []string = make([]string, len(servers))
|
||||||
|
|
||||||
for i, server := range servers {
|
for i, server := range servers {
|
||||||
// Check if the server is in the valid server list passed at startup
|
// Check if the server is in the valid server list passed at startup
|
||||||
var isValidServer bool = false
|
var isValidServer bool = false
|
||||||
for _, validServer := range settingServers {
|
for _, validServer := range settingServers {
|
||||||
if validServer == server {
|
if validServer == server {
|
||||||
isValidServer = true
|
isValidServer = true
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if !isValidServer {
|
if !isValidServer {
|
||||||
// If the server is not valid, create a dummy goroutine to return a failure
|
// If the server is not valid, create a dummy goroutine to return a failure
|
||||||
go func (i int) {
|
go func(i int) {
|
||||||
ch <- channelData{i, "request failed: invalid server\n"}
|
ch <- channelData{i, "request failed: invalid server\n"}
|
||||||
} (i)
|
}(i)
|
||||||
} else {
|
} else {
|
||||||
// Compose URL and send the request
|
// Compose URL and send the request
|
||||||
url := "http://" + server + "." + settingServersDomain + ":" + strconv.Itoa(settingServersPort) + "/" + url.PathEscape(endpoint) + "?q=" + url.QueryEscape(command)
|
url := "http://" + server + "." + settingServersDomain + ":" + strconv.Itoa(settingServersPort) + "/" + url.PathEscape(endpoint) + "?q=" + url.QueryEscape(command)
|
||||||
go func (url string, i int){
|
go func(url string, i int) {
|
||||||
response, err := http.Get(url)
|
response, err := http.Get(url)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ch <- channelData{i, "request failed: " + err.Error() + "\n"}
|
ch <- channelData{i, "request failed: " + err.Error() + "\n"}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
text, _ := ioutil.ReadAll(response.Body)
|
text, _ := ioutil.ReadAll(response.Body)
|
||||||
ch <- channelData{i, string(text)}
|
if len(text) == 0 {
|
||||||
} (url, i)
|
text = []byte("node returned empty response, please refresh to try again.")
|
||||||
}
|
}
|
||||||
}
|
ch <- channelData{i, string(text)}
|
||||||
|
}(url, i)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Sort the responses by their ids, to return data in order
|
// Sort the responses by their ids, to return data in order
|
||||||
for range servers {
|
for range servers {
|
||||||
var output channelData = <-ch
|
var output channelData = <-ch
|
||||||
response_array[output.id] = output.data
|
responseArray[output.id] = output.data
|
||||||
}
|
}
|
||||||
|
|
||||||
return response_array
|
return responseArray
|
||||||
}
|
}
|
||||||
|
@ -1,144 +1,106 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strconv"
|
||||||
"strconv"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Helper to check if the IP is valid
|
// Helper to check if the IP is valid
|
||||||
func isIP(s string) bool {
|
func isIP(s string) bool {
|
||||||
return nil != net.ParseIP(s)
|
return nil != net.ParseIP(s)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Helper to check if the number is valid
|
// Helper to check if the number is valid
|
||||||
func isNumber(s string) bool {
|
func isNumber(s string) bool {
|
||||||
_, err := strconv.Atoi(s)
|
_, err := strconv.Atoi(s)
|
||||||
return nil == err
|
return nil == err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Print HTML header to the given http response
|
// Print HTML header to the given http response
|
||||||
func templateHeader(w http.ResponseWriter, r *http.Request, title string) {
|
func templateHeader(w http.ResponseWriter, r *http.Request, title string) {
|
||||||
path := r.URL.Path
|
path := r.URL.Path[1:]
|
||||||
split := strings.Split(r.URL.Path, "/")
|
split := strings.Split(path, "/")
|
||||||
|
|
||||||
// Mark if the URL is for a whois query
|
// Mark if the URL is for a whois query
|
||||||
var isWhois bool = false
|
var isWhois bool = split[0] == "whois"
|
||||||
if len(split) >= 2 && split[1] == "whois" {
|
var whoisTarget string = strings.Join(split[1:], "/")
|
||||||
isWhois = true
|
|
||||||
}
|
|
||||||
|
|
||||||
// Use a default URL if the request URL is too short
|
// Use a default URL if the request URL is too short
|
||||||
// The URL is for return to IPv4 summary page
|
// The URL is for return to IPv4 summary page
|
||||||
if len(split) < 4 {
|
if len(split) < 3 {
|
||||||
path = "/ipv4/summary/" + strings.Join(settingServers[:], "+") + "/"
|
path = "ipv4/summary/" + strings.Join(settingServers[:], "+") + "/"
|
||||||
} else if len(split) == 4 {
|
} else if len(split) == 3 {
|
||||||
path += "/"
|
path += "/"
|
||||||
}
|
}
|
||||||
|
|
||||||
// Compose URLs for link in navbar
|
split = strings.Split(path, "/")
|
||||||
split = strings.Split(path, "/")
|
|
||||||
split[1] = "ipv4"
|
|
||||||
ipv4_url := strings.Join(split, "/")
|
|
||||||
|
|
||||||
split = strings.Split(path, "/")
|
// Compose URLs for link in navbar
|
||||||
split[1] = "ipv6"
|
ipv4URL := "/" + strings.Join([]string{"ipv4", split[1], split[2], strings.Join(split[3:], "/")}, "/")
|
||||||
ipv6_url := strings.Join(split, "/")
|
ipv6URL := "/" + strings.Join([]string{"ipv6", split[1], split[2], strings.Join(split[3:], "/")}, "/")
|
||||||
|
allURL := "/" + strings.Join([]string{split[0], split[1], strings.Join(settingServers[:], "+"), strings.Join(split[3:], "/")}, "/")
|
||||||
|
|
||||||
split = strings.Split(path, "/")
|
// Check if the "All Server" link should be marked as active
|
||||||
split[3] = strings.Join(settingServers[:], "+")
|
var serverAllActive bool = strings.ToLower(split[2]) == strings.ToLower(strings.Join(settingServers[:], "+"))
|
||||||
all_url := strings.Join(split, "/")
|
|
||||||
|
|
||||||
// Check if the "All Server" link should be marked as active
|
// Print the IPv4, IPv6, All Servers link in navbar
|
||||||
split = strings.Split(path, "/")
|
var serverNavigation string = `
|
||||||
var serverAllActive string
|
<li class="nav-item"><a class="nav-link" href="` + ipv4URL + `"` + (map[bool]string{true: " active"})[strings.ToLower(split[0]) == "ipv4"] + `> IPv4 </a></li>
|
||||||
if split[3] == strings.Join(settingServers[:], "+") {
|
<li class="nav-item"><a class="nav-link" href="` + ipv6URL + `"` + (map[bool]string{true: " active"})[strings.ToLower(split[0]) == "ipv6"] + `> IPv6 </a></li>
|
||||||
serverAllActive = " active"
|
|
||||||
}
|
|
||||||
|
|
||||||
// Print the IPv4, IPv6, All Servers link in navbar
|
|
||||||
var serverNavigation string = `
|
|
||||||
<li class="nav-item">
|
|
||||||
<a class="nav-link" href="` + ipv4_url + `"> IPv4 </a>
|
|
||||||
</li>
|
|
||||||
<li class="nav-item">
|
|
||||||
<a class="nav-link" href="` + ipv6_url + `"> IPv6 </a>
|
|
||||||
</li>
|
|
||||||
<span class="navbar-text">|</span>
|
<span class="navbar-text">|</span>
|
||||||
<li class="nav-item">
|
<li class="nav-item">
|
||||||
<a class="nav-link` + serverAllActive + `" href="` + all_url + `"> All Servers </a>
|
<a class="nav-link` + (map[bool]string{true: " active"})[serverAllActive] + `" href="` + allURL + `"> All Servers </a>
|
||||||
</li>
|
</li>`
|
||||||
`
|
|
||||||
|
|
||||||
// Add a link for each of the servers
|
// Add a link for each of the servers
|
||||||
for _, server := range settingServers {
|
for _, server := range settingServers {
|
||||||
split = strings.Split(path, "/")
|
var serverActive string
|
||||||
var serverActive string
|
if split[2] == server {
|
||||||
if split[3] == server {
|
serverActive = " active"
|
||||||
serverActive = " active"
|
}
|
||||||
}
|
serverURL := "/" + strings.Join([]string{split[0], split[1], server, strings.Join(split[3:], "/")}, "/")
|
||||||
split[3] = server
|
|
||||||
server_url := strings.Join(split, "/")
|
|
||||||
|
|
||||||
serverNavigation += `
|
serverNavigation += `
|
||||||
<li class="nav-item">
|
<li class="nav-item">
|
||||||
<a class="nav-link` + serverActive + `" href="` + server_url + `">` + server + `</a>
|
<a class="nav-link` + serverActive + `" href="` + serverURL + `">` + server + `</a>
|
||||||
</li>
|
</li>`
|
||||||
`
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// Add the options in navbar form, and check if they are active
|
// Add the options in navbar form, and check if they are active
|
||||||
var options string
|
var optionKeys = []string{"summary", "detail", "route", "route_all", "route_where", "route_where_all", "whois", "traceroute"}
|
||||||
split = strings.Split(path, "/")
|
var optionDisplays = []string{
|
||||||
if split[2] == "summary" {
|
"show protocol",
|
||||||
options += `<option value="summary" selected>show protocol</option>`
|
"show protocol all",
|
||||||
} else {
|
"show route for ...",
|
||||||
options += `<option value="summary">show protocol</option>`
|
"show route for ... all",
|
||||||
}
|
"show route where net ~ [ ... ]",
|
||||||
if split[2] == "route" {
|
"show route where net ~ [ ... ] all",
|
||||||
options += `<option value="route" selected>show route for ...</option>`
|
"whois ...",
|
||||||
} else {
|
"traceroute ...",
|
||||||
options += `<option value="route">show route for ...</option>`
|
}
|
||||||
}
|
|
||||||
if split[2] == "route_all" {
|
|
||||||
options += `<option value="route_all" selected>show route for ... all</option>`
|
|
||||||
} else {
|
|
||||||
options += `<option value="route_all">show route for ... all</option>`
|
|
||||||
}
|
|
||||||
if split[2] == "route_where" {
|
|
||||||
options += `<option value="route_where" selected>show route where net ~ [ ... ]</option>`
|
|
||||||
} else {
|
|
||||||
options += `<option value="route_where">show route where net ~ [ ... ]</option>`
|
|
||||||
}
|
|
||||||
if split[2] == "route_where_all" {
|
|
||||||
options += `<option value="route_where_all" selected>show route where net ~ [ ... ] all</option>`
|
|
||||||
} else {
|
|
||||||
options += `<option value="route_where_all">show route where net ~ [ ... ] all</option>`
|
|
||||||
}
|
|
||||||
if isWhois {
|
|
||||||
options += `<option value="whois" selected>whois ...</option>`
|
|
||||||
} else {
|
|
||||||
options += `<option value="whois">whois ...</option>`
|
|
||||||
}
|
|
||||||
if split[2] == "traceroute" {
|
|
||||||
options += `<option value="traceroute" selected>traceroute ...</option>`
|
|
||||||
} else {
|
|
||||||
options += `<option value="traceroute">traceroute ...</option>`
|
|
||||||
}
|
|
||||||
|
|
||||||
var target string
|
var options string
|
||||||
if isWhois {
|
for optionKeyID, optionKey := range optionKeys {
|
||||||
// This is a whois request, use original path URL instead of the modified one
|
options += "<option value=\"" + optionKey + "\""
|
||||||
// and extract the target
|
if (optionKey == "whois" && isWhois) || optionKey == split[1] {
|
||||||
whoisSplit := strings.Split(r.URL.Path, "/")
|
options += " selected"
|
||||||
target = strings.Join(whoisSplit[2:], "/")
|
}
|
||||||
} else if len(split) >= 5 {
|
options += ">" + optionDisplays[optionKeyID] + "</option>"
|
||||||
// This is a normal request, just extract the target
|
}
|
||||||
target = strings.Join(split[4:], "/")
|
|
||||||
}
|
|
||||||
|
|
||||||
w.Write([]byte(`
|
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="zh-CN" lang="zh-CN" class="no-js">
|
||||||
<head>
|
<head>
|
||||||
@ -162,8 +124,8 @@ func templateHeader(w http.ResponseWriter, r *http.Request, title string) {
|
|||||||
<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">` + options + `</select>
|
||||||
<input name="proto" class="d-none" value="` + split[1] + `">
|
<input name="proto" class="d-none" value="` + split[0] + `">
|
||||||
<input name="server" class="d-none" value="` + split[3] + `">
|
<input name="server" class="d-none" value="` + split[2] + `">
|
||||||
<input name="target" class="form-control" placeholder="Target" aria-label="Target" value="` + target + `">
|
<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>
|
||||||
@ -179,7 +141,7 @@ func templateHeader(w http.ResponseWriter, r *http.Request, title string) {
|
|||||||
|
|
||||||
// Print HTML footer to http response
|
// Print HTML footer to http response
|
||||||
func templateFooter(w http.ResponseWriter) {
|
func templateFooter(w http.ResponseWriter) {
|
||||||
w.Write([]byte(`
|
w.Write([]byte(`
|
||||||
</div>
|
</div>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
@ -189,120 +151,112 @@ func templateFooter(w http.ResponseWriter) {
|
|||||||
// Write the given text to http response, and add whois links for
|
// Write the given text to http response, and add whois links for
|
||||||
// ASNs and IP addresses
|
// ASNs and IP addresses
|
||||||
func smartWriter(w http.ResponseWriter, s string) {
|
func smartWriter(w http.ResponseWriter, s string) {
|
||||||
w.Write([]byte("<pre>"))
|
w.Write([]byte("<pre>"))
|
||||||
for _, line := range strings.Split(s, "\n") {
|
for _, line := range strings.Split(s, "\n") {
|
||||||
var tabPending bool = false
|
var isASes bool = false
|
||||||
var isFirstWord bool = true
|
|
||||||
var isASes bool = false
|
|
||||||
for _, word := range strings.Split(line, " ") {
|
|
||||||
// Process each word
|
|
||||||
if len(word) == 0 {
|
|
||||||
// Indicates that two spaces are connected together
|
|
||||||
// Replace this with a tab later
|
|
||||||
tabPending = true
|
|
||||||
} else {
|
|
||||||
if isFirstWord {
|
|
||||||
// Do not add space before the first word
|
|
||||||
isFirstWord = false
|
|
||||||
} else if tabPending {
|
|
||||||
// A tab should be added; add it
|
|
||||||
w.Write([]byte("\t"))
|
|
||||||
tabPending = false
|
|
||||||
} else {
|
|
||||||
// Two words separated by a space, just print the space
|
|
||||||
w.Write([]byte(" "))
|
|
||||||
}
|
|
||||||
|
|
||||||
if isIP(word) {
|
var lineFormatted string
|
||||||
// Add whois link to the IP, handles IPv4 and IPv6
|
words := strings.Split(line, " ")
|
||||||
w.Write([]byte("<a href=\"/whois/" + word + "\">" + word + "</a>"))
|
|
||||||
} else if len(strings.Split(word, "%")) == 2 && isIP(strings.Split(word, "%")[0]) {
|
for wordID, word := range words {
|
||||||
// IPv6 link-local with interface name, like fd00::1%eth0
|
if len(word) == 0 {
|
||||||
// Add whois link to address part
|
continue
|
||||||
w.Write([]byte("<a href=\"/whois/" + strings.Split(word, "%")[0] + "\">" + strings.Split(word, "%")[0] + "</a>"))
|
}
|
||||||
w.Write([]byte("%" + strings.Split(word, "%")[1]))
|
if wordID > 0 && (len(words[wordID-1]) == 0 || words[wordID-1][len(words[wordID-1])-1] == ':') {
|
||||||
} else if len(strings.Split(word, "/")) == 2 && isIP(strings.Split(word, "/")[0]) {
|
// Insert TAB if there are multiple spaces before this word
|
||||||
// IP with a CIDR range, like 192.168.0.1/24
|
lineFormatted += "\t"
|
||||||
// Add whois link to first part
|
} else {
|
||||||
w.Write([]byte("<a href=\"/whois/" + strings.Split(word, "/")[0] + "\">" + strings.Split(word, "/")[0] + "</a>"))
|
lineFormatted += " "
|
||||||
w.Write([]byte("/" + strings.Split(word, "/")[1]))
|
}
|
||||||
} else if word == "AS:" || word == "\tBGP.as_path:" {
|
|
||||||
// Bird will output ASNs later
|
if isIP(word) {
|
||||||
isASes = true
|
// Add whois link to the IP, handles IPv4 and IPv6
|
||||||
w.Write([]byte(word))
|
lineFormatted += "<a href=\"/whois/" + word + "\">" + word + "</a>"
|
||||||
} else if isASes && isNumber(word) {
|
} else if len(strings.Split(word, "%")) == 2 && isIP(strings.Split(word, "%")[0]) {
|
||||||
// Bird is outputing ASNs, ass whois for them
|
// IPv6 link-local with interface name, like fd00::1%eth0
|
||||||
w.Write([]byte("<a href=\"/whois/AS" + word + "\">" + word + "</a>"))
|
// Add whois link to address part
|
||||||
} else {
|
lineFormatted += "<a href=\"/whois/" + strings.Split(word, "%")[0] + "\">" + strings.Split(word, "%")[0] + "</a>"
|
||||||
// Just an ordinary word, print it and done
|
lineFormatted += "%" + strings.Split(word, "%")[1]
|
||||||
w.Write([]byte(word))
|
} 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>"
|
||||||
w.Write([]byte("\n"))
|
lineFormatted += "/" + strings.Split(word, "/")[1]
|
||||||
}
|
} else if word == "AS:" || word == "\tBGP.as_path:" {
|
||||||
w.Write([]byte("</pre>"))
|
// Bird will output ASNs later
|
||||||
|
isASes = true
|
||||||
|
lineFormatted += word
|
||||||
|
} else if isASes && isNumber(word) {
|
||||||
|
// Bird is outputing ASNs, ass whois for them
|
||||||
|
lineFormatted += "<a href=\"/whois/AS" + word + "\">" + 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
|
// Output a table for the summary page
|
||||||
func summaryTable(w http.ResponseWriter, isIPv6 bool, data string, serverName string) {
|
func summaryTable(w http.ResponseWriter, isIPv6 bool, data string, serverName string) {
|
||||||
w.Write([]byte("<table class=\"table table-striped table-bordered table-sm\">"))
|
// w.Write([]byte("<pre>" + data + "</pre>"))
|
||||||
for lineId, line := range strings.Split(data, "\n") {
|
w.Write([]byte("<table class=\"table table-striped table-bordered table-sm\">"))
|
||||||
var tabPending bool = false
|
for lineID, line := range strings.Split(data, "\n") {
|
||||||
var tableCells int = 0
|
var row [6]string
|
||||||
var row [6]string
|
var rowIndex int = 0
|
||||||
for i, word := range strings.Split(line, " ") {
|
|
||||||
if len(word) == 0 {
|
|
||||||
tabPending = true
|
|
||||||
} else {
|
|
||||||
if i == 0 {
|
|
||||||
tabPending = true
|
|
||||||
} else if tabPending {
|
|
||||||
// Allow up to 6 columns in the table, any more is ignored
|
|
||||||
if tableCells < 5 {
|
|
||||||
tableCells++
|
|
||||||
} else {
|
|
||||||
row[tableCells] += " "
|
|
||||||
}
|
|
||||||
tabPending = false
|
|
||||||
} else {
|
|
||||||
row[tableCells] += " "
|
|
||||||
}
|
|
||||||
row[tableCells] += word
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ignore empty lines
|
words := strings.Split(line, " ")
|
||||||
if len(row[0]) == 0 {
|
for wordID, word := range words {
|
||||||
continue
|
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
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if lineId == 0 {
|
// Ignore empty lines
|
||||||
// Draw the table head
|
if len(row[0]) == 0 {
|
||||||
w.Write([]byte("<thead>"))
|
continue
|
||||||
for i := 0; i < 6; i++ {
|
}
|
||||||
w.Write([]byte("<th scope=\"col\">" + row[i] + "</th>"))
|
|
||||||
}
|
if lineID == 0 {
|
||||||
w.Write([]byte("</thead><tbody>"))
|
// Draw the table head
|
||||||
} else {
|
w.Write([]byte("<thead>"))
|
||||||
// Draw the row in red if the link isn't up
|
for i := 0; i < 6; i++ {
|
||||||
if row[3] == "up" {
|
w.Write([]byte("<th scope=\"col\">" + row[i] + "</th>"))
|
||||||
w.Write([]byte("<tr>"))
|
}
|
||||||
} else if lineId != 0 {
|
w.Write([]byte("</thead><tbody>"))
|
||||||
w.Write([]byte("<tr class=\"table-danger\">"))
|
} else {
|
||||||
}
|
// Draw the row in red if the link isn't up
|
||||||
// Add link to detail for first column
|
w.Write([]byte("<tr class=\"" + (map[string]string{
|
||||||
if isIPv6 {
|
"up": "table-success",
|
||||||
w.Write([]byte("<td><a href=\"/ipv6/detail/" + serverName + "/" + row[0] + "\">" + row[0] + "</a></td>"))
|
"down": "table-danger",
|
||||||
} else {
|
"start": "table-danger",
|
||||||
w.Write([]byte("<td><a href=\"/ipv4/detail/" + serverName + "/" + row[0] + "\">" + row[0] + "</a></td>"))
|
})[row[3]] + "\">"))
|
||||||
}
|
// Add link to detail for first column
|
||||||
// Draw the other cells
|
if isIPv6 {
|
||||||
for i := 1; i < 6; i++ {
|
w.Write([]byte("<td><a href=\"/ipv6/detail/" + serverName + "/" + row[0] + "\">" + row[0] + "</a></td>"))
|
||||||
w.Write([]byte("<td>" + row[i] + "</td>"))
|
} else {
|
||||||
}
|
w.Write([]byte("<td><a href=\"/ipv4/detail/" + serverName + "/" + row[0] + "\">" + row[0] + "</a></td>"))
|
||||||
w.Write([]byte("</tr>"))
|
}
|
||||||
}
|
// Draw the other cells
|
||||||
}
|
for i := 1; i < 6; i++ {
|
||||||
w.Write([]byte("</tbody></table>"))
|
w.Write([]byte("<td>" + row[i] + "</td>"))
|
||||||
|
}
|
||||||
|
w.Write([]byte("</tr>"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
w.Write([]byte("</tbody></table>"))
|
||||||
}
|
}
|
||||||
|
@ -1,144 +1,87 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"html"
|
||||||
"strings"
|
"net/http"
|
||||||
"html"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
func webDispatcherIPv4Summary(w http.ResponseWriter, r *http.Request) {
|
func webHandlerWhois(w http.ResponseWriter, r *http.Request) {
|
||||||
split := strings.Split(r.URL.Path[len("/ipv4/summary/"):], "/")
|
var target string = r.URL.Path[len("/whois/"):]
|
||||||
webHandler(w, r, "bird", split[0], "show protocols")
|
|
||||||
|
templateHeader(w, r, "Bird-lg Go - whois "+html.EscapeString(target))
|
||||||
|
|
||||||
|
w.Write([]byte("<h2>whois " + html.EscapeString(target) + "</h2>"))
|
||||||
|
smartWriter(w, whois(target))
|
||||||
|
|
||||||
|
templateFooter(w)
|
||||||
}
|
}
|
||||||
|
|
||||||
func webDispatcherIPv6Summary(w http.ResponseWriter, r *http.Request) {
|
func webBackendCommunicator(w http.ResponseWriter, r *http.Request, endpoint string, command string) {
|
||||||
split := strings.Split(r.URL.Path[len("/ipv6/summary/"):], "/")
|
split := strings.Split(r.URL.Path[1:], "/")
|
||||||
webHandler(w, r, "bird6", split[0], "show protocols")
|
urlCommands := strings.Join(split[3:], "/")
|
||||||
|
|
||||||
|
command = (map[string]string{
|
||||||
|
"summary": "show protocols",
|
||||||
|
"detail": "show protocols all " + urlCommands,
|
||||||
|
"route": "show route for " + urlCommands,
|
||||||
|
"route_all": "show route for " + urlCommands + " all",
|
||||||
|
"route_where": "show route where net ~ [ " + urlCommands + " ]",
|
||||||
|
"route_where_all": "show route where net ~ [ " + urlCommands + " ] all",
|
||||||
|
"traceroute": urlCommands,
|
||||||
|
})[command]
|
||||||
|
|
||||||
|
templateHeader(w, r, "Bird-lg Go - "+html.EscapeString(endpoint+" "+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)
|
||||||
}
|
}
|
||||||
|
|
||||||
func webDispatcherIPv4Detail(w http.ResponseWriter, r *http.Request) {
|
func webHandlerNavbarFormRedirect(w http.ResponseWriter, r *http.Request) {
|
||||||
split := strings.Split(r.URL.Path[len("/ipv4/detail/"):], "/")
|
query := r.URL.Query()
|
||||||
webHandler(w, r, "bird", split[0], "show protocols all " + split[1])
|
if query.Get("action") == "whois" {
|
||||||
}
|
http.Redirect(w, r, "/"+query.Get("action")+"/"+query.Get("target"), 302)
|
||||||
|
} else if query.Get("action") == "summary" {
|
||||||
func webDispatcherIPv6Detail(w http.ResponseWriter, r *http.Request) {
|
http.Redirect(w, r, "/"+query.Get("proto")+"/"+query.Get("action")+"/"+query.Get("server"), 302)
|
||||||
split := strings.Split(r.URL.Path[len("/ipv6/detail/"):], "/")
|
} else {
|
||||||
webHandler(w, r, "bird6", split[0], "show protocols all " + split[1])
|
http.Redirect(w, r, "/"+query.Get("proto")+"/"+query.Get("action")+"/"+query.Get("server")+"/"+query.Get("target"), 302)
|
||||||
}
|
}
|
||||||
|
|
||||||
func webDispatcherIPv4Route(w http.ResponseWriter, r *http.Request) {
|
|
||||||
split := strings.Split(r.URL.Path[len("/ipv4/route/"):], "/")
|
|
||||||
webHandler(w, r, "bird", split[0], "show route for " + strings.Join(split[1:], "/"))
|
|
||||||
}
|
|
||||||
|
|
||||||
func webDispatcherIPv6Route(w http.ResponseWriter, r *http.Request) {
|
|
||||||
split := strings.Split(r.URL.Path[len("/ipv6/route/"):], "/")
|
|
||||||
webHandler(w, r, "bird6", split[0], "show route for " + strings.Join(split[1:], "/"))
|
|
||||||
}
|
|
||||||
|
|
||||||
func webDispatcherIPv4RouteAll(w http.ResponseWriter, r *http.Request) {
|
|
||||||
split := strings.Split(r.URL.Path[len("/ipv4/route_all/"):], "/")
|
|
||||||
webHandler(w, r, "bird", split[0], "show route for " + strings.Join(split[1:], "/") + " all")
|
|
||||||
}
|
|
||||||
|
|
||||||
func webDispatcherIPv6RouteAll(w http.ResponseWriter, r *http.Request) {
|
|
||||||
split := strings.Split(r.URL.Path[len("/ipv6/route_all/"):], "/")
|
|
||||||
webHandler(w, r, "bird6", split[0], "show route for " + strings.Join(split[1:], "/") + " all")
|
|
||||||
}
|
|
||||||
|
|
||||||
func webDispatcherIPv4RouteWhere(w http.ResponseWriter, r *http.Request) {
|
|
||||||
split := strings.Split(r.URL.Path[len("/ipv4/route_where/"):], "/")
|
|
||||||
webHandler(w, r, "bird", split[0], "show route where net ~ [ " + strings.Join(split[1:], "/") + " ]")
|
|
||||||
}
|
|
||||||
|
|
||||||
func webDispatcherIPv6RouteWhere(w http.ResponseWriter, r *http.Request) {
|
|
||||||
split := strings.Split(r.URL.Path[len("/ipv6/route_where/"):], "/")
|
|
||||||
webHandler(w, r, "bird6", split[0], "show route where net ~ [ " + strings.Join(split[1:], "/") + " ]")
|
|
||||||
}
|
|
||||||
|
|
||||||
func webDispatcherIPv4RouteWhereAll(w http.ResponseWriter, r *http.Request) {
|
|
||||||
split := strings.Split(r.URL.Path[len("/ipv4/route_where_all/"):], "/")
|
|
||||||
webHandler(w, r, "bird", split[0], "show route where net ~ [ " + strings.Join(split[1:], "/") + " ] all")
|
|
||||||
}
|
|
||||||
|
|
||||||
func webDispatcherIPv6RouteWhereAll(w http.ResponseWriter, r *http.Request) {
|
|
||||||
split := strings.Split(r.URL.Path[len("/ipv6/route_where_all/"):], "/")
|
|
||||||
webHandler(w, r, "bird6", split[0], "show route where net ~ [ " + strings.Join(split[1:], "/") + " ] all")
|
|
||||||
}
|
|
||||||
|
|
||||||
func webDispatcherWhois(w http.ResponseWriter, r *http.Request) {
|
|
||||||
var target string = r.URL.Path[len("/whois/"):]
|
|
||||||
|
|
||||||
templateHeader(w, r, "Bird-lg Go - whois " + html.EscapeString(target))
|
|
||||||
|
|
||||||
w.Write([]byte("<h2>whois " + html.EscapeString(target) + "</h2>"))
|
|
||||||
smartWriter(w, whois(target))
|
|
||||||
|
|
||||||
templateFooter(w)
|
|
||||||
}
|
|
||||||
|
|
||||||
func webDispatcherIPv4Traceroute(w http.ResponseWriter, r *http.Request) {
|
|
||||||
split := strings.Split(r.URL.Path[len("/ipv4/traceroute/"):], "/")
|
|
||||||
webHandler(w, r, "traceroute", split[0], strings.Join(split[1:], "/"))
|
|
||||||
}
|
|
||||||
|
|
||||||
func webDispatcherIPv6Traceroute(w http.ResponseWriter, r *http.Request) {
|
|
||||||
split := strings.Split(r.URL.Path[len("/ipv6/traceroute/"):], "/")
|
|
||||||
webHandler(w, r, "traceroute6", split[0], strings.Join(split[1:], "/"))
|
|
||||||
}
|
|
||||||
|
|
||||||
func webHandler(w http.ResponseWriter, r *http.Request, endpoint string, serverQuery string, command string) {
|
|
||||||
templateHeader(w, r, "Bird-lg Go - " + html.EscapeString(endpoint + " " + command))
|
|
||||||
|
|
||||||
var servers []string = strings.Split(serverQuery, "+")
|
|
||||||
|
|
||||||
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" && 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 defaultRedirect(w http.ResponseWriter, r *http.Request) {
|
|
||||||
http.Redirect(w, r, "/ipv4/summary/" + strings.Join(settingServers[:], "+"), 302)
|
|
||||||
}
|
|
||||||
|
|
||||||
func navbarFormRedirect(w http.ResponseWriter, r *http.Request) {
|
|
||||||
query := r.URL.Query()
|
|
||||||
if query.Get("action") == "whois" {
|
|
||||||
http.Redirect(w, r, "/" + query.Get("action") + "/" + query.Get("target"), 302)
|
|
||||||
} else if query.Get("action") == "summary" {
|
|
||||||
http.Redirect(w, r, "/" + query.Get("proto") + "/" + query.Get("action") + "/" + query.Get("server"), 302)
|
|
||||||
} else {
|
|
||||||
http.Redirect(w, r, "/" + query.Get("proto") + "/" + query.Get("action") + "/" + query.Get("server") + "/" + query.Get("target"), 302)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func webServerStart() {
|
func webServerStart() {
|
||||||
// Start HTTP server
|
// Start HTTP server
|
||||||
http.HandleFunc("/", defaultRedirect)
|
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
|
||||||
http.HandleFunc("/ipv4/summary/", webDispatcherIPv4Summary)
|
http.Redirect(w, r, "/ipv4/summary/"+strings.Join(settingServers[:], "+"), 302)
|
||||||
http.HandleFunc("/ipv6/summary/", webDispatcherIPv6Summary)
|
})
|
||||||
http.HandleFunc("/ipv4/detail/", webDispatcherIPv4Detail)
|
http.HandleFunc("/ipv4/summary/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "summary") })
|
||||||
http.HandleFunc("/ipv6/detail/", webDispatcherIPv6Detail)
|
http.HandleFunc("/ipv6/summary/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "summary") })
|
||||||
http.HandleFunc("/ipv4/route/", webDispatcherIPv4Route)
|
http.HandleFunc("/ipv4/detail/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "detail") })
|
||||||
http.HandleFunc("/ipv6/route/", webDispatcherIPv6Route)
|
http.HandleFunc("/ipv6/detail/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "detail") })
|
||||||
http.HandleFunc("/ipv4/route_all/", webDispatcherIPv4RouteAll)
|
http.HandleFunc("/ipv4/route/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "route") })
|
||||||
http.HandleFunc("/ipv6/route_all/", webDispatcherIPv6RouteAll)
|
http.HandleFunc("/ipv6/route/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "route") })
|
||||||
http.HandleFunc("/ipv4/route_where/", webDispatcherIPv4RouteWhere)
|
http.HandleFunc("/ipv4/route_all/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "route_all") })
|
||||||
http.HandleFunc("/ipv6/route_where/", webDispatcherIPv6RouteWhere)
|
http.HandleFunc("/ipv6/route_all/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "route_all") })
|
||||||
http.HandleFunc("/ipv4/route_where_all/", webDispatcherIPv4RouteWhereAll)
|
http.HandleFunc("/ipv4/route_where/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "route_where") })
|
||||||
http.HandleFunc("/ipv6/route_where_all/", webDispatcherIPv6RouteWhereAll)
|
http.HandleFunc("/ipv6/route_where/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "route_where") })
|
||||||
http.HandleFunc("/ipv4/traceroute/", webDispatcherIPv4Traceroute)
|
http.HandleFunc("/ipv4/route_where_all/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird", "route_where_all") })
|
||||||
http.HandleFunc("/ipv6/traceroute/", webDispatcherIPv6Traceroute)
|
http.HandleFunc("/ipv6/route_where_all/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "bird6", "route_where_all") })
|
||||||
http.HandleFunc("/whois/", webDispatcherWhois)
|
http.HandleFunc("/ipv4/traceroute/", func(w http.ResponseWriter, r *http.Request) { webBackendCommunicator(w, r, "traceroute", "traceroute") })
|
||||||
http.HandleFunc("/redir/", navbarFormRedirect)
|
http.HandleFunc("/ipv6/traceroute/", func(w http.ResponseWriter, r *http.Request) {
|
||||||
http.ListenAndServe(settingListen, nil)
|
webBackendCommunicator(w, r, "traceroute6", "traceroute")
|
||||||
|
})
|
||||||
|
http.HandleFunc("/whois/", webHandlerWhois)
|
||||||
|
http.HandleFunc("/redir/", webHandlerNavbarFormRedirect)
|
||||||
|
http.ListenAndServe(settingListen, nil)
|
||||||
}
|
}
|
||||||
|
144
proxy/bird.go
144
proxy/bird.go
@ -1,10 +1,10 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"sync"
|
"sync"
|
||||||
)
|
)
|
||||||
|
|
||||||
// BIRDv4 connection & mutex lock
|
// BIRDv4 connection & mutex lock
|
||||||
@ -18,83 +18,93 @@ var bird6Mutex = &sync.Mutex{}
|
|||||||
// Read a line from bird socket, removing preceding status number, output it.
|
// Read a line from bird socket, removing preceding status number, output it.
|
||||||
// Returns if there are more lines.
|
// Returns if there are more lines.
|
||||||
func birdReadln(bird io.Reader, w io.Writer) bool {
|
func birdReadln(bird io.Reader, w io.Writer) bool {
|
||||||
// Read from socket byte by byte, until reaching newline character
|
// Read from socket byte by byte, until reaching newline character
|
||||||
c := make([]byte, 1024, 1024)
|
c := make([]byte, 1024, 1024)
|
||||||
pos := 0
|
pos := 0
|
||||||
for {
|
for {
|
||||||
if pos >= 1024 { break }
|
if pos >= 1024 {
|
||||||
_, err := bird.Read(c[pos:pos+1])
|
break
|
||||||
if err != nil {
|
}
|
||||||
panic(err)
|
_, err := bird.Read(c[pos : pos+1])
|
||||||
}
|
if err != nil {
|
||||||
if c[pos] == byte('\n') {
|
panic(err)
|
||||||
break
|
}
|
||||||
}
|
if c[pos] == byte('\n') {
|
||||||
pos++
|
break
|
||||||
}
|
}
|
||||||
|
pos++
|
||||||
|
}
|
||||||
|
|
||||||
c = c[:pos+1]
|
c = c[:pos+1]
|
||||||
// print(string(c[:]))
|
// print(string(c[:]))
|
||||||
|
|
||||||
// Remove preceding status number, different situations
|
// Remove preceding status number, different situations
|
||||||
if pos < 4 {
|
if pos < 4 {
|
||||||
// Line is too short to have a status number
|
// Line is too short to have a status number
|
||||||
if w != nil {
|
if w != nil {
|
||||||
pos = 0
|
pos = 0
|
||||||
for c[pos] == byte(' ') { pos++ }
|
for c[pos] == byte(' ') {
|
||||||
w.Write(c[pos:])
|
pos++
|
||||||
}
|
}
|
||||||
return true
|
w.Write(c[pos:])
|
||||||
} else if isNumeric(c[0]) && isNumeric(c[1]) && isNumeric(c[2]) && isNumeric(c[3]) {
|
}
|
||||||
// There is a status number at beginning, remove first 5 bytes
|
return true
|
||||||
if w != nil && pos > 6 {
|
} else if isNumeric(c[0]) && isNumeric(c[1]) && isNumeric(c[2]) && isNumeric(c[3]) {
|
||||||
pos = 5
|
// There is a status number at beginning, remove first 5 bytes
|
||||||
for c[pos] == byte(' ') { pos++ }
|
if w != nil && pos > 6 {
|
||||||
w.Write(c[pos:])
|
pos = 5
|
||||||
}
|
for c[pos] == byte(' ') {
|
||||||
return c[0] != byte('0') && c[0] != byte('8') && c[0] != byte('9')
|
pos++
|
||||||
} else {
|
}
|
||||||
// There is no status number, only remove preceding spaces
|
w.Write(c[pos:])
|
||||||
if w != nil {
|
}
|
||||||
pos = 0
|
return c[0] != byte('0') && c[0] != byte('8') && c[0] != byte('9')
|
||||||
for c[pos] == byte(' ') { pos++ }
|
} else {
|
||||||
w.Write(c[pos:])
|
// There is no status number, only remove preceding spaces
|
||||||
}
|
if w != nil {
|
||||||
return true
|
pos = 0
|
||||||
}
|
for c[pos] == byte(' ') {
|
||||||
|
pos++
|
||||||
|
}
|
||||||
|
w.Write(c[pos:])
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Write a command to a bird socket
|
// Write a command to a bird socket
|
||||||
func birdWriteln(bird io.Writer, s string) {
|
func birdWriteln(bird io.Writer, s string) {
|
||||||
bird.Write([]byte(s + "\n"))
|
bird.Write([]byte(s + "\n"))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handles BIRDv4 queries
|
// Handles BIRDv4 queries
|
||||||
func birdHandler(httpW http.ResponseWriter, httpR *http.Request) {
|
func birdHandler(httpW http.ResponseWriter, httpR *http.Request) {
|
||||||
query := string(httpR.URL.Query().Get("q"))
|
query := string(httpR.URL.Query().Get("q"))
|
||||||
if query == "" {
|
if query == "" {
|
||||||
invalidHandler(httpW, httpR)
|
invalidHandler(httpW, httpR)
|
||||||
} else {
|
} else {
|
||||||
birdMutex.Lock()
|
birdMutex.Lock()
|
||||||
defer birdMutex.Unlock()
|
defer birdMutex.Unlock()
|
||||||
|
|
||||||
println(query)
|
println(query)
|
||||||
birdWriteln(bird, query)
|
birdWriteln(bird, query)
|
||||||
for birdReadln(bird, httpW) {}
|
for birdReadln(bird, httpW) {
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handles BIRDv6 queries
|
// Handles BIRDv6 queries
|
||||||
func bird6Handler(httpW http.ResponseWriter, httpR *http.Request) {
|
func bird6Handler(httpW http.ResponseWriter, httpR *http.Request) {
|
||||||
query := string(httpR.URL.Query().Get("q"))
|
query := string(httpR.URL.Query().Get("q"))
|
||||||
if query == "" {
|
if query == "" {
|
||||||
invalidHandler(httpW, httpR)
|
invalidHandler(httpW, httpR)
|
||||||
} else {
|
} else {
|
||||||
bird6Mutex.Lock()
|
bird6Mutex.Lock()
|
||||||
defer bird6Mutex.Unlock()
|
defer bird6Mutex.Unlock()
|
||||||
|
|
||||||
println(query)
|
println(query)
|
||||||
birdWriteln(bird6, query)
|
birdWriteln(bird6, query)
|
||||||
for birdReadln(bird6, httpW) {}
|
for birdReadln(bird6, httpW) {
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
102
proxy/main.go
102
proxy/main.go
@ -1,75 +1,75 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net"
|
"flag"
|
||||||
"net/http"
|
"net"
|
||||||
"flag"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Check if a byte is character for number
|
// Check if a byte is character for number
|
||||||
func isNumeric(b byte) bool {
|
func isNumeric(b byte) bool {
|
||||||
return b >= byte('0') && b <= byte('9')
|
return b >= byte('0') && b <= byte('9')
|
||||||
}
|
}
|
||||||
|
|
||||||
// Default handler, returns 500 Internal Server Error
|
// Default handler, returns 500 Internal Server Error
|
||||||
func invalidHandler(httpW http.ResponseWriter, httpR *http.Request) {
|
func invalidHandler(httpW http.ResponseWriter, httpR *http.Request) {
|
||||||
httpW.WriteHeader(http.StatusInternalServerError)
|
httpW.WriteHeader(http.StatusInternalServerError)
|
||||||
httpW.Write([]byte("Invalid Request\n"))
|
httpW.Write([]byte("Invalid Request\n"))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wrapper of tracer
|
// Wrapper of tracer
|
||||||
func main() {
|
func main() {
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
// Prepare default socket paths, use environment variable if possible
|
// Prepare default socket paths, use environment variable if possible
|
||||||
birdSocketDefault := "/var/run/bird/bird.ctl"
|
birdSocketDefault := "/var/run/bird/bird.ctl"
|
||||||
bird6SocketDefault := "/var/run/bird/bird6.ctl"
|
bird6SocketDefault := "/var/run/bird/bird6.ctl"
|
||||||
listenDefault := ":8000"
|
listenDefault := ":8000"
|
||||||
|
|
||||||
if birdSocketEnv := os.Getenv("BIRD_SOCKET"); birdSocketEnv != "" {
|
if birdSocketEnv := os.Getenv("BIRD_SOCKET"); birdSocketEnv != "" {
|
||||||
birdSocketDefault = birdSocketEnv
|
birdSocketDefault = birdSocketEnv
|
||||||
}
|
}
|
||||||
if bird6SocketEnv := os.Getenv("BIRD6_SOCKET"); bird6SocketEnv != "" {
|
if bird6SocketEnv := os.Getenv("BIRD6_SOCKET"); bird6SocketEnv != "" {
|
||||||
bird6SocketDefault = bird6SocketEnv
|
bird6SocketDefault = bird6SocketEnv
|
||||||
}
|
}
|
||||||
if listenEnv := os.Getenv("BIRDLG_LISTEN"); listenEnv != "" {
|
if listenEnv := os.Getenv("BIRDLG_LISTEN"); listenEnv != "" {
|
||||||
listenDefault = listenEnv
|
listenDefault = listenEnv
|
||||||
}
|
}
|
||||||
|
|
||||||
// Allow parameters to override environment variables
|
// Allow parameters to override environment variables
|
||||||
birdParam := flag.String("bird", birdSocketDefault, "socket file for bird, set either in parameter or environment variable BIRD_SOCKET")
|
birdParam := flag.String("bird", birdSocketDefault, "socket file for bird, set either in parameter or environment variable BIRD_SOCKET")
|
||||||
bird6Param := flag.String("bird6", bird6SocketDefault, "socket file for bird6, set either in parameter or environment variable BIRD6_SOCKET")
|
bird6Param := flag.String("bird6", bird6SocketDefault, "socket file for bird6, set either in parameter or environment variable BIRD6_SOCKET")
|
||||||
listenParam := flag.String("listen", listenDefault, "listen address, set either in parameter or environment variable BIRDLG_LISTEN")
|
listenParam := flag.String("listen", listenDefault, "listen address, set either in parameter or environment variable BIRDLG_LISTEN")
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
|
||||||
// Initialize BIRDv4 socket
|
// Initialize BIRDv4 socket
|
||||||
bird, err = net.Dial("unix", *birdParam)
|
bird, err = net.Dial("unix", *birdParam)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
defer bird.Close()
|
defer bird.Close()
|
||||||
|
|
||||||
birdReadln(bird, nil)
|
birdReadln(bird, nil)
|
||||||
birdWriteln(bird, "restrict")
|
birdWriteln(bird, "restrict")
|
||||||
birdReadln(bird, nil)
|
birdReadln(bird, nil)
|
||||||
|
|
||||||
// Initialize BIRDv6 socket
|
// Initialize BIRDv6 socket
|
||||||
bird6, err = net.Dial("unix", *bird6Param)
|
bird6, err = net.Dial("unix", *bird6Param)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
defer bird6.Close()
|
defer bird6.Close()
|
||||||
|
|
||||||
birdReadln(bird6, nil)
|
birdReadln(bird6, nil)
|
||||||
birdWriteln(bird6, "restrict")
|
birdWriteln(bird6, "restrict")
|
||||||
birdReadln(bird6, nil)
|
birdReadln(bird6, nil)
|
||||||
|
|
||||||
// Start HTTP server
|
// Start HTTP server
|
||||||
http.HandleFunc("/", invalidHandler)
|
http.HandleFunc("/", invalidHandler)
|
||||||
http.HandleFunc("/bird", birdHandler)
|
http.HandleFunc("/bird", birdHandler)
|
||||||
http.HandleFunc("/bird6", bird6Handler)
|
http.HandleFunc("/bird6", bird6Handler)
|
||||||
http.HandleFunc("/traceroute", tracerouteIPv4Wrapper)
|
http.HandleFunc("/traceroute", tracerouteIPv4Wrapper)
|
||||||
http.HandleFunc("/traceroute6", tracerouteIPv6Wrapper)
|
http.HandleFunc("/traceroute6", tracerouteIPv6Wrapper)
|
||||||
http.ListenAndServe(*listenParam, nil)
|
http.ListenAndServe(*listenParam, nil)
|
||||||
}
|
}
|
||||||
|
@ -1,67 +1,75 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"runtime"
|
"os/exec"
|
||||||
"os/exec"
|
"runtime"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Wrapper of traceroute, IPv4
|
// Wrapper of traceroute, IPv4
|
||||||
func tracerouteIPv4Wrapper(httpW http.ResponseWriter, httpR *http.Request) {
|
func tracerouteIPv4Wrapper(httpW http.ResponseWriter, httpR *http.Request) {
|
||||||
tracerouteRealHandler(false, httpW, httpR)
|
tracerouteRealHandler(false, httpW, httpR)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wrapper of traceroute, IPv6
|
// Wrapper of traceroute, IPv6
|
||||||
func tracerouteIPv6Wrapper(httpW http.ResponseWriter, httpR *http.Request) {
|
func tracerouteIPv6Wrapper(httpW http.ResponseWriter, httpR *http.Request) {
|
||||||
tracerouteRealHandler(true, httpW, httpR)
|
tracerouteRealHandler(true, httpW, httpR)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Real handler of traceroute requests
|
// Real handler of traceroute requests
|
||||||
func tracerouteRealHandler(useIPv6 bool, httpW http.ResponseWriter, httpR *http.Request) {
|
func tracerouteRealHandler(useIPv6 bool, httpW http.ResponseWriter, httpR *http.Request) {
|
||||||
query := string(httpR.URL.Query().Get("q"))
|
query := string(httpR.URL.Query().Get("q"))
|
||||||
if query == "" {
|
if query == "" {
|
||||||
invalidHandler(httpW, httpR)
|
invalidHandler(httpW, httpR)
|
||||||
} else {
|
} else {
|
||||||
var cmd string
|
var cmd string
|
||||||
var args []string
|
var args []string
|
||||||
if runtime.GOOS == "freebsd" || runtime.GOOS == "netbsd" {
|
if runtime.GOOS == "freebsd" || runtime.GOOS == "netbsd" {
|
||||||
if useIPv6 { cmd = "traceroute6" } else { cmd = "traceroute" }
|
if useIPv6 {
|
||||||
args = []string{"-a", "-q1", "-w1", "-m15", query}
|
cmd = "traceroute6"
|
||||||
} else if runtime.GOOS == "openbsd" {
|
} else {
|
||||||
if useIPv6 { cmd = "traceroute6" } else { cmd = "traceroute" }
|
cmd = "traceroute"
|
||||||
args = []string{"-A", "-q1", "-w1", "-m15", query}
|
}
|
||||||
} else if runtime.GOOS == "linux" {
|
args = []string{"-a", "-q1", "-w1", "-m15", query}
|
||||||
cmd = "traceroute"
|
} else if runtime.GOOS == "openbsd" {
|
||||||
if useIPv6 {
|
if useIPv6 {
|
||||||
args = []string{"-6", "-A", "-q1", "-N32", "-w1", "-m15", query}
|
cmd = "traceroute6"
|
||||||
} else {
|
} else {
|
||||||
args = []string{"-4", "-A", "-q1", "-N32", "-w1", "-m15", query}
|
cmd = "traceroute"
|
||||||
}
|
}
|
||||||
} else {
|
args = []string{"-A", "-q1", "-w1", "-m15", query}
|
||||||
httpW.WriteHeader(http.StatusInternalServerError)
|
} else if runtime.GOOS == "linux" {
|
||||||
httpW.Write([]byte("Traceroute Not Supported\n"))
|
cmd = "traceroute"
|
||||||
return
|
if useIPv6 {
|
||||||
}
|
args = []string{"-6", "-A", "-q1", "-N32", "-w1", "-m15", query}
|
||||||
instance := exec.Command(cmd, args...)
|
} else {
|
||||||
output, err := instance.Output()
|
args = []string{"-4", "-A", "-q1", "-N32", "-w1", "-m15", query}
|
||||||
if err != nil && runtime.GOOS == "linux" {
|
}
|
||||||
// Standard traceroute utility failed, maybe system using busybox
|
} else {
|
||||||
// Run with less parameters
|
httpW.WriteHeader(http.StatusInternalServerError)
|
||||||
cmd = "traceroute"
|
httpW.Write([]byte("Traceroute Not Supported\n"))
|
||||||
if useIPv6 {
|
return
|
||||||
args = []string{"-6", "-q1", "-w1", "-m15", query}
|
}
|
||||||
} else {
|
instance := exec.Command(cmd, args...)
|
||||||
args = []string{"-4", "-q1", "-w1", "-m15", query}
|
output, err := instance.Output()
|
||||||
}
|
if err != nil && runtime.GOOS == "linux" {
|
||||||
instance = exec.Command(cmd, args...)
|
// Standard traceroute utility failed, maybe system using busybox
|
||||||
output, err = instance.Output()
|
// Run with less parameters
|
||||||
}
|
cmd = "traceroute"
|
||||||
if err != nil {
|
if useIPv6 {
|
||||||
httpW.WriteHeader(http.StatusInternalServerError)
|
args = []string{"-6", "-q1", "-w1", "-m15", query}
|
||||||
httpW.Write([]byte("Traceroute Execution Error: "))
|
} else {
|
||||||
httpW.Write([]byte(err.Error() + "\n"))
|
args = []string{"-4", "-q1", "-w1", "-m15", query}
|
||||||
return
|
}
|
||||||
}
|
instance = exec.Command(cmd, args...)
|
||||||
httpW.Write(output)
|
output, err = instance.Output()
|
||||||
}
|
}
|
||||||
|
if err != nil {
|
||||||
|
httpW.WriteHeader(http.StatusInternalServerError)
|
||||||
|
httpW.Write([]byte("Traceroute Execution Error: "))
|
||||||
|
httpW.Write([]byte(err.Error() + "\n"))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
httpW.Write(output)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user