Remove BIRDv1 support

This commit is contained in:
Lan Tian 2020-10-30 23:10:03 +08:00
parent 79431effb2
commit 3bcfc3d36c
No known key found for this signature in database
GPG Key ID: 3D2E9DC81E5791C7
12 changed files with 141 additions and 229 deletions

View File

@ -3,6 +3,8 @@ Bird-lg-go
An alternative implementation for [bird-lg](https://github.com/sileht/bird-lg) written in Go. Both frontend and backend (proxy) are implemented, and can work with either the original Python implementation or the Go implementation. An alternative implementation for [bird-lg](https://github.com/sileht/bird-lg) written in Go. Both frontend and backend (proxy) are implemented, and can work with either the original Python implementation or the Go implementation.
> The code on master branch no longer support BIRDv1. Branch "bird1" is the last version that supports BIRDv1.
Frontend Frontend
-------- --------
@ -18,11 +20,16 @@ Features implemented:
Usage: all configuration is done via commandline parameters or environment variables, no config file. Usage: all configuration is done via commandline parameters or environment variables, no config file.
- --servers / BIRDLG_SERVERS: server name prefixes, separated by comma | Parameter | Environment Variable | Description |
- --domain / BIRDLG_DOMAIN: server name domain suffixes | --------- | -------------------- | ----------- |
- --listen / BIRDLG_LISTEN: address bird-lg is listening on (default ":5000") | --servers | BIRDLG_SERVERS | server name prefixes, separated by comma |
- --proxy-port / BIRDLG_PROXY_PORT: port bird-lgproxy is running on (default 8000) | --domain | BIRDLG_DOMAIN | server name domain suffixes |
- --whois / BIRDLG_WHOIS: whois server for queries (default "whois.verisign-grs.com") | --listen | BIRDLG_LISTEN | address bird-lg is listening on (default ":5000") |
| --proxy-port | BIRDLG_PROXY_PORT | port bird-lgproxy is running on (default 8000) |
| --whois | BIRDLG_WHOIS | whois server for queries (default "whois.verisign-grs.com") |
| --dns-interface | BIRDLG_DNS_INTERFACE | dns zone to query ASN information (default "asn.cymru.com") |
| --title-brand | BIRDLG_TITLE_BRAND | prefix of page titles in browser tabs (default "Bird-lg Go") |
| --navbar-brand | BIRDLG_NAVBAR_BRAND | brand to show in the navigation bar (default "Bird-lg Go") |
Example: the following command starts the frontend with 2 BIRD nodes, with domain name "gigsgigscloud.dn42.lantian.pub" and "hostdare.dn42.lantian.pub", and proxies are running on port 8000 on both nodes. Example: the following command starts the frontend with 2 BIRD nodes, with domain name "gigsgigscloud.dn42.lantian.pub" and "hostdare.dn42.lantian.pub", and proxies are running on port 8000 on both nodes.
@ -50,18 +57,18 @@ The proxy directory contains the code for the "proxy" for bird commands and trac
Features implemented: Features implemented:
- Sending queries to BIRD and BIRD6 - Sending queries to BIRD
- If you are using BIRDv2, simply point both `--bird` and `--bird6` to the only socket file of BIRDv2
- Sending "restrict" command to BIRD to prevent unauthorized changes - Sending "restrict" command to BIRD to prevent unauthorized changes
- Executing traceroute command on Linux, FreeBSD and OpenBSD - Executing traceroute command on Linux, FreeBSD and OpenBSD
- Source IP restriction - Source IP restriction
Usage: all configuration is done via commandline parameters or environment variables, no config file. Usage: all configuration is done via commandline parameters or environment variables, no config file.
- --allowed / ALLOWED_IPS: IPs allowed to access this proxy, separated by commas. Don't set to allow all IPs. (default "") | Parameter | Environment Variable | Description |
- --bird / BIRD_SOCKET: socket file for bird, set either in parameter or environment variable BIRD_SOCKET (default "/var/run/bird/bird.ctl") | --------- | -------------------- | ----------- |
- --bird6 / BIRD6_SOCKET: socket file for bird6, set either in parameter or environment variable BIRD6_SOCKET (default "/var/run/bird/bird6.ctl") | --allowed | ALLOWED_IPS | IPs allowed to access this proxy, separated by commas. Don't set to allow all IPs. (default "") |
- --listen / BIRDLG_LISTEN: listen address, set either in parameter or environment variable BIRDLG_LISTEN (default ":8000") | --bird | BIRD_SOCKET | socket file for bird, set either in parameter or environment variable BIRD_SOCKET (default "/var/run/bird/bird.ctl") |
| --listen | BIRDLG_LISTEN | listen address, set either in parameter or environment variable BIRDLG_LISTEN (default ":8000") |
Example: start proxy with default configuration, should work "out of the box" on Debian 9 with BIRDv1: Example: start proxy with default configuration, should work "out of the box" on Debian 9 with BIRDv1:
@ -69,7 +76,7 @@ Example: start proxy with default configuration, should work "out of the box" on
Example: start proxy with custom bird socket location: Example: start proxy with custom bird socket location:
./proxy --bird /run/bird.ctl --bird6 /run/bird6.ctl ./proxy --bird /run/bird.ctl
Example: the following docker-compose.yml entry does the same as above, but by starting a Docker container: Example: the following docker-compose.yml entry does the same as above, but by starting a Docker container:
@ -79,7 +86,6 @@ Example: the following docker-compose.yml entry does the same as above, but by s
restart: always restart: always
volumes: volumes:
- "/run/bird.ctl:/var/run/bird/bird.ctl" - "/run/bird.ctl:/var/run/bird/bird.ctl"
- "/run/bird6.ctl:/var/run/bird/bird6.ctl"
ports: ports:
- "192.168.0.1:8000:8000" - "192.168.0.1:8000:8000"
@ -88,6 +94,7 @@ You can use source IP restriction to increase security. You should also bind the
Credits Credits
------- -------
- Everyone who contributed to this project (see Contributors section on the right)
- Mehdi Abaakouk for creating [the original bird-lg project](https://github.com/sileht/bird-lg) - Mehdi Abaakouk for creating [the original bird-lg project](https://github.com/sileht/bird-lg)
- [Bootstrap](https://getbootstrap.com/) as web UI framework - [Bootstrap](https://getbootstrap.com/) as web UI framework
@ -95,4 +102,3 @@ License
------- -------
GPL 3.0 GPL 3.0

5
frontend/go.mod Normal file
View File

@ -0,0 +1,5 @@
module github.com/xddxdd/bird-lg-go/frontend
go 1.15
require github.com/gorilla/handlers v1.5.1

4
frontend/go.sum Normal file
View File

@ -0,0 +1,4 @@
github.com/felixge/httpsnoop v1.0.1 h1:lvB5Jl89CsZtGIWuTcDM1E/vkVs49/Ml7JJe07l8SPQ=
github.com/felixge/httpsnoop v1.0.1/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U=
github.com/gorilla/handlers v1.5.1 h1:9lRY6j8DEeeBT10CvO9hGW0gmky0BprnvDI5vfhUHH4=
github.com/gorilla/handlers v1.5.1/go.mod h1:t8XrUpc4KVXb7HGyJ4/cEnwQiaxrX/hz1Zv/4g96P1Q=

View File

@ -9,20 +9,20 @@ import (
func renderTemplate(w http.ResponseWriter, r *http.Request, title string, content string) { func renderTemplate(w http.ResponseWriter, r *http.Request, title string, content string) {
path := r.URL.Path[1:] path := r.URL.Path[1:]
split := strings.SplitN(path, "/", 4) split := strings.SplitN(path, "/", 3)
isWhois := strings.ToLower(split[0]) == "whois" isWhois := strings.ToLower(split[0]) == "whois"
whoisTarget := strings.Join(split[1:], "/") whoisTarget := strings.Join(split[1:], "/")
// 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 summary page
if len(split) < 3 { if len(split) < 2 {
path = "ipv4/summary/" + strings.Join(setting.servers, "+") + "/" path = "summary/" + strings.Join(setting.servers, "+") + "/"
} else if len(split) == 3 { } else if len(split) == 2 {
path += "/" path += "/"
} }
split = strings.SplitN(path, "/", 4) split = strings.SplitN(path, "/", 3)
var args tmplArguments var args tmplArguments
args.Options = map[string]string{ args.Options = map[string]string{
@ -40,24 +40,20 @@ func renderTemplate(w http.ResponseWriter, r *http.Request, title string, conten
"traceroute": "traceroute ...", "traceroute": "traceroute ...",
} }
args.Servers = setting.servers args.Servers = setting.servers
args.AllServersLinkActive = strings.ToLower(split[2]) == strings.ToLower(strings.Join(setting.servers, "+")) args.AllServersLinkActive = strings.ToLower(split[1]) == strings.ToLower(strings.Join(setting.servers, "+"))
args.AllServersURL = strings.Join(setting.servers, "+") args.AllServersURL = strings.Join(setting.servers, "+")
args.IsWhois = isWhois args.IsWhois = isWhois
args.WhoisTarget = whoisTarget args.WhoisTarget = whoisTarget
args.URLProto = strings.ToLower(split[0]) args.URLOption = strings.ToLower(split[0])
args.URLOption = strings.ToLower(split[1]) args.URLServer = strings.ToLower(split[1])
args.URLServer = strings.ToLower(split[2]) args.URLCommand = split[2]
args.URLCommand = split[3]
args.Title = setting.titleBrand + title args.Title = setting.titleBrand + title
args.Brand = setting.navBarBrand args.Brand = setting.navBarBrand
args.Content = content args.Content = content
err := tmpl.Execute(w, args) tmpl.Execute(w, args)
if err != nil {
panic(err)
}
} }
// Write the given text to http response, and add whois links for // Write the given text to http response, and add whois links for
@ -87,7 +83,7 @@ type summaryTableArguments struct {
} }
// Output a table for the summary page // Output a table for the summary page
func summaryTable(isIPv6 bool, data string, serverName string) string { func summaryTable(data string, serverName string) string {
var result string var result string
// Sort the table, excluding title row // Sort the table, excluding title row
@ -152,11 +148,7 @@ func summaryTable(isIPv6 bool, data string, serverName string) string {
"passive": "table-info", "passive": "table-info",
})[row[3]] + `">` })[row[3]] + `">`
// Add link to detail for first column // Add link to detail for first column
if isIPv6 { result += `<td><a href="/detail/` + serverName + `/` + row[0] + `">` + row[0] + `</a></td>`
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 // Draw the other cells
for i := 1; i < 6; i++ { for i := 1; i < 6; i++ {
result += "<td>" + row[i] + "</td>" result += "<td>" + row[i] + "</td>"

View File

@ -87,17 +87,13 @@ func webHandlerTelegramBot(w http.ResponseWriter, r *http.Request) {
commandResult := "" commandResult := ""
// - traceroute // - traceroute
if telegramIsCommand(request.Message.Text, "trace") || telegramIsCommand(request.Message.Text, "trace4") { if telegramIsCommand(request.Message.Text, "trace") {
commandResult = telegramBatchRequestFormat(servers, "traceroute", target, telegramDefaultPostProcess) commandResult = telegramBatchRequestFormat(servers, "traceroute", target, telegramDefaultPostProcess)
} else if telegramIsCommand(request.Message.Text, "trace6") {
commandResult = telegramBatchRequestFormat(servers, "traceroute6", target, telegramDefaultPostProcess)
} else if telegramIsCommand(request.Message.Text, "route") || telegramIsCommand(request.Message.Text, "route4") { } else if telegramIsCommand(request.Message.Text, "route") {
commandResult = telegramBatchRequestFormat(servers, "bird", "show route for "+target+" primary", telegramDefaultPostProcess) commandResult = telegramBatchRequestFormat(servers, "bird", "show route for "+target+" primary", telegramDefaultPostProcess)
} else if telegramIsCommand(request.Message.Text, "route6") {
commandResult = telegramBatchRequestFormat(servers, "bird6", "show route for "+target+" primary", telegramDefaultPostProcess)
} else if telegramIsCommand(request.Message.Text, "path") || telegramIsCommand(request.Message.Text, "path4") { } else if telegramIsCommand(request.Message.Text, "path") {
commandResult = telegramBatchRequestFormat(servers, "bird", "show route for "+target+" all primary", func(result string) string { commandResult = telegramBatchRequestFormat(servers, "bird", "show route for "+target+" all primary", func(result string) string {
for _, s := range strings.Split(result, "\n") { for _, s := range strings.Split(result, "\n") {
if strings.Contains(s, "BGP.as_path: ") { if strings.Contains(s, "BGP.as_path: ") {
@ -106,15 +102,6 @@ func webHandlerTelegramBot(w http.ResponseWriter, r *http.Request) {
} }
return "" return ""
}) })
} else if telegramIsCommand(request.Message.Text, "path6") {
commandResult = telegramBatchRequestFormat(servers, "bird6", "show route for "+target+" all primary", func(result string) string {
for _, s := range strings.Split(result, "\n") {
if strings.Contains(s, "BGP.as_path: ") {
return strings.TrimSpace(strings.Split(s, ":")[1])
}
}
return ""
})
} else if telegramIsCommand(request.Message.Text, "whois") { } else if telegramIsCommand(request.Message.Text, "whois") {
if setting.netSpecificMode == "dn42" { if setting.netSpecificMode == "dn42" {
@ -137,9 +124,9 @@ func webHandlerTelegramBot(w http.ResponseWriter, r *http.Request) {
} else if telegramIsCommand(request.Message.Text, "help") { } else if telegramIsCommand(request.Message.Text, "help") {
commandResult = ` commandResult = `
/[path|path6] <IP> /path <IP>
/[route|route6] <IP> /route <IP>
/[trace|trace6] <IP> /trace <IP>
/whois <Target> /whois <Target>
` `
} else { } else {

View File

@ -17,7 +17,6 @@ type tmplArguments struct {
IsWhois bool IsWhois bool
WhoisTarget string WhoisTarget string
URLProto string
URLOption string URLOption string
URLServer string URLServer string
URLCommand string URLCommand string
@ -49,34 +48,37 @@ var tmpl = template.Must(template.New("tmpl").Parse(`
</button> </button>
<div class="collapse navbar-collapse" id="navbarSupportedContent"> <div class="collapse navbar-collapse" id="navbarSupportedContent">
{{ $option := .URLOption }}
{{ $server := .URLServer }}
{{ $target := .URLCommand }}
{{ if .IsWhois }}
{{ $option = "summary" }}
{{ $server = .AllServersURL }}
{{ $target = "" }}
{{ end }}
<ul class="navbar-nav mr-auto"> <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"> <li class="nav-item">
<a class="nav-link{{ if .AllServersLinkActive }} active{{ end }}" href="/{{ .URLProto }}/{{ .URLOption }}/{{ .AllServersURL }}/{{ .URLCommand }}"> All Servers </a> <a class="nav-link{{ if .AllServersLinkActive }} active{{ end }}"
href="/{{ $option }}/{{ .AllServersURL }}/{{ $target }}"> All Servers </a>
</li> </li>
{{ range $k, $v := .Servers }} {{ range $k, $v := .Servers }}
<li class="nav-item"> <li class="nav-item">
<a class="nav-link{{ if eq $.URLServer $v }} active{{ end }}" href="/{{ $.URLProto }}/{{ $.URLOption }}/{{ $v }}/{{ $.URLCommand }}">{{ $v }}</a> <a class="nav-link{{ if eq $server $v }} active{{ end }}"
href="/{{ $option }}/{{ $v }}/{{ $target }}">{{ $v }}</a>
</li> </li>
{{ end }} {{ end }}
</ul> </ul>
{{ $option := .URLOption }}
{{ $target := .URLCommand }}
{{ if .IsWhois }} {{ if .IsWhois }}
{{ $option = "whois" }}
{{ $target = .WhoisTarget }} {{ $target = .WhoisTarget }}
{{ end }} {{ 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"> <select name="action" class="form-control">
{{ range $k, $v := .Options }} {{ range $k, $v := .Options }}
<option value="{{ $k }}"{{ if eq $k $option }} selected{{end}}>{{ $v }}</option> <option value="{{ $k }}"{{ if eq $k $.URLOption }} selected{{end}}>{{ $v }}</option>
{{ end }} {{ end }}
</select> </select>
<input name="proto" class="d-none" value="{{ .URLProto }}"> <input name="server" class="d-none" value="{{ $server }}">
<input name="server" class="d-none" value="{{ .URLServer }}">
<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">&raquo;</button> <button class="btn btn-outline-success" type="submit">&raquo;</button>

View File

@ -40,8 +40,8 @@ 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.SplitN(r.URL.Path[1:], "/", 4) split := strings.SplitN(r.URL.Path[1:], "/", 4)
var urlCommands string var urlCommands string
if len(split) >= 4 { if len(split) >= 3 {
urlCommands = split[3] urlCommands = split[2]
} }
var backendCommand string var backendCommand string
@ -52,14 +52,13 @@ func webBackendCommunicator(endpoint string, command string) func(w http.Respons
} }
backendCommand = strings.TrimSpace(backendCommand) backendCommand = strings.TrimSpace(backendCommand)
var servers []string = strings.Split(split[2], "+") var servers []string = strings.Split(split[1], "+")
var responses []string = batchRequest(servers, endpoint, backendCommand) var responses []string = batchRequest(servers, endpoint, backendCommand)
var result string var result string
for i, response := range responses { for i, response := range responses {
result += "<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") && backendCommand == "show protocols" && len(response) > 4 && strings.ToLower(response[0:4]) == "name" {
var isIPv6 bool = endpoint[len(endpoint)-1] == '6' result += summaryTable(response, servers[i])
result += summaryTable(isIPv6, response, servers[i])
} else { } else {
result += smartFormatter(response) result += smartFormatter(response)
} }
@ -85,7 +84,7 @@ func webHandlerBGPMap(endpoint string, command string) func(w http.ResponseWrite
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.Split(r.URL.Path[1:], "/")
urlCommands := strings.Join(split[3:], "/") urlCommands := strings.Join(split[2:], "/")
var backendCommand string var backendCommand string
if strings.Contains(backendCommandPrimitive, "%") { if strings.Contains(backendCommandPrimitive, "%") {
@ -94,7 +93,7 @@ func webHandlerBGPMap(endpoint string, command string) func(w http.ResponseWrite
backendCommand = backendCommandPrimitive backendCommand = backendCommandPrimitive
} }
var servers []string = strings.Split(split[2], "+") var servers []string = strings.Split(split[1], "+")
var responses []string = batchRequest(servers, endpoint, backendCommand) var responses []string = batchRequest(servers, endpoint, backendCommand)
renderTemplate( renderTemplate(
w, r, w, r,
@ -121,41 +120,41 @@ func webHandlerNavbarFormRedirect(w http.ResponseWriter, r *http.Request) {
if query.Get("action") == "whois" { if query.Get("action") == "whois" {
http.Redirect(w, r, "/"+query.Get("action")+"/"+query.Get("target"), 302) http.Redirect(w, r, "/"+query.Get("action")+"/"+query.Get("target"), 302)
} else if query.Get("action") == "summary" { } else if query.Get("action") == "summary" {
http.Redirect(w, r, "/"+query.Get("proto")+"/"+query.Get("action")+"/"+query.Get("server"), 302) http.Redirect(w, r, "/"+query.Get("action")+"/"+query.Get("server")+"/", 302)
} else { } else {
http.Redirect(w, r, "/"+query.Get("proto")+"/"+query.Get("action")+"/"+query.Get("server")+"/"+query.Get("target"), 302) http.Redirect(w, r, "/"+query.Get("action")+"/"+query.Get("server")+"/"+query.Get("target"), 302)
} }
} }
func webHandlerRobotsTxt(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("User-agent: *\nDisallow: /\n"))
}
func webHandler404(w http.ResponseWriter, r *http.Request) {
w.WriteHeader(http.StatusNotFound)
w.Write([]byte("404 not found\n"))
}
func webServerStart() { func webServerStart() {
// Start HTTP server // Start HTTP server
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(setting.servers, "+"), 302) http.Redirect(w, r, "/summary/"+strings.Join(setting.servers, "+"), 302)
}) })
http.HandleFunc("/ipv4/summary/", webBackendCommunicator("bird", "summary")) http.HandleFunc("/summary/", webBackendCommunicator("bird", "summary"))
http.HandleFunc("/ipv6/summary/", webBackendCommunicator("bird6", "summary")) http.HandleFunc("/detail/", webBackendCommunicator("bird", "detail"))
http.HandleFunc("/ipv4/detail/", webBackendCommunicator("bird", "detail")) http.HandleFunc("/route/", webBackendCommunicator("bird", "route"))
http.HandleFunc("/ipv6/detail/", webBackendCommunicator("bird6", "detail")) http.HandleFunc("/route_all/", webBackendCommunicator("bird", "route_all"))
http.HandleFunc("/ipv4/route/", webBackendCommunicator("bird", "route")) http.HandleFunc("/route_bgpmap/", webHandlerBGPMap("bird", "route_bgpmap"))
http.HandleFunc("/ipv6/route/", webBackendCommunicator("bird6", "route")) http.HandleFunc("/route_where/", webBackendCommunicator("bird", "route_where"))
http.HandleFunc("/ipv4/route_all/", webBackendCommunicator("bird", "route_all")) http.HandleFunc("/route_where_all/", webBackendCommunicator("bird", "route_where_all"))
http.HandleFunc("/ipv6/route_all/", webBackendCommunicator("bird6", "route_all")) http.HandleFunc("/route_where_bgpmap/", webHandlerBGPMap("bird", "route_where_bgpmap"))
http.HandleFunc("/ipv4/route_bgpmap/", webHandlerBGPMap("bird", "route_bgpmap")) http.HandleFunc("/route_generic/", webBackendCommunicator("bird", "route_generic"))
http.HandleFunc("/ipv6/route_bgpmap/", webHandlerBGPMap("bird6", "route_bgpmap")) http.HandleFunc("/generic/", webBackendCommunicator("bird", "generic"))
http.HandleFunc("/ipv4/route_where/", webBackendCommunicator("bird", "route_where")) http.HandleFunc("/traceroute/", webBackendCommunicator("traceroute", "traceroute"))
http.HandleFunc("/ipv6/route_where/", webBackendCommunicator("bird6", "route_where"))
http.HandleFunc("/ipv4/route_where_all/", webBackendCommunicator("bird", "route_where_all"))
http.HandleFunc("/ipv6/route_where_all/", webBackendCommunicator("bird6", "route_where_all"))
http.HandleFunc("/ipv4/route_where_bgpmap/", webHandlerBGPMap("bird", "route_where_bgpmap"))
http.HandleFunc("/ipv6/route_where_bgpmap/", webHandlerBGPMap("bird6", "route_where_bgpmap"))
http.HandleFunc("/ipv4/route_generic/", webBackendCommunicator("bird", "route_generic"))
http.HandleFunc("/ipv6/route_generic/", webBackendCommunicator("bird6", "route_generic"))
http.HandleFunc("/ipv4/generic/", webBackendCommunicator("bird", "generic"))
http.HandleFunc("/ipv6/generic/", webBackendCommunicator("bird6", "generic"))
http.HandleFunc("/ipv4/traceroute/", webBackendCommunicator("traceroute", "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.HandleFunc("/robots.txt", webHandlerRobotsTxt)
http.HandleFunc("/favicon.ico", webHandler404)
http.ListenAndServe(setting.listen, handlers.LoggingHandler(os.Stdout, http.DefaultServeMux)) http.ListenAndServe(setting.listen, handlers.LoggingHandler(os.Stdout, http.DefaultServeMux))
} }

View File

@ -89,25 +89,3 @@ func birdHandler(httpW http.ResponseWriter, httpR *http.Request) {
} }
} }
} }
// Handles BIRDv6 queries
func bird6Handler(httpW http.ResponseWriter, httpR *http.Request) {
query := string(httpR.URL.Query().Get("q"))
if query == "" {
invalidHandler(httpW, httpR)
} else {
// Initialize BIRDv6 socket
bird6, err := net.Dial("unix", setting.bird6Socket)
if err != nil {
panic(err)
}
defer bird6.Close()
birdReadln(bird6, nil)
birdWriteln(bird6, "restrict")
birdReadln(bird6, nil)
birdWriteln(bird6, query)
for birdReadln(bird6, httpW) {
}
}
}

5
proxy/go.mod Normal file
View File

@ -0,0 +1,5 @@
module github.com/xddxdd/bird-lg-go/proxy
go 1.15
require github.com/gorilla/handlers v1.5.1

4
proxy/go.sum Normal file
View File

@ -0,0 +1,4 @@
github.com/felixge/httpsnoop v1.0.1 h1:lvB5Jl89CsZtGIWuTcDM1E/vkVs49/Ml7JJe07l8SPQ=
github.com/felixge/httpsnoop v1.0.1/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U=
github.com/gorilla/handlers v1.5.1 h1:9lRY6j8DEeeBT10CvO9hGW0gmky0BprnvDI5vfhUHH4=
github.com/gorilla/handlers v1.5.1/go.mod h1:t8XrUpc4KVXb7HGyJ4/cEnwQiaxrX/hz1Zv/4g96P1Q=

View File

@ -50,10 +50,9 @@ func accessHandler(next http.Handler) http.Handler {
} }
type settingType struct { type settingType struct {
birdSocket string birdSocket string
bird6Socket string listen string
listen string allowedIPs []string
allowedIPs []string
} }
var setting settingType var setting settingType
@ -63,7 +62,6 @@ func main() {
// Prepare default socket paths, use environment variable if possible // Prepare default socket paths, use environment variable if possible
var settingDefault = settingType{ var settingDefault = settingType{
"/var/run/bird/bird.ctl", "/var/run/bird/bird.ctl",
"/var/run/bird/bird6.ctl",
":8000", ":8000",
[]string{""}, []string{""},
} }
@ -71,9 +69,6 @@ func main() {
if birdSocketEnv := os.Getenv("BIRD_SOCKET"); birdSocketEnv != "" { if birdSocketEnv := os.Getenv("BIRD_SOCKET"); birdSocketEnv != "" {
settingDefault.birdSocket = birdSocketEnv settingDefault.birdSocket = birdSocketEnv
} }
if bird6SocketEnv := os.Getenv("BIRD6_SOCKET"); bird6SocketEnv != "" {
settingDefault.bird6Socket = bird6SocketEnv
}
if listenEnv := os.Getenv("BIRDLG_LISTEN"); listenEnv != "" { if listenEnv := os.Getenv("BIRDLG_LISTEN"); listenEnv != "" {
settingDefault.listen = listenEnv settingDefault.listen = listenEnv
} }
@ -83,21 +78,17 @@ func main() {
// Allow parameters to override environment variables // Allow parameters to override environment variables
birdParam := flag.String("bird", settingDefault.birdSocket, "socket file for bird, set either in parameter or environment variable BIRD_SOCKET") birdParam := flag.String("bird", settingDefault.birdSocket, "socket file for bird, set either in parameter or environment variable BIRD_SOCKET")
bird6Param := flag.String("bird6", settingDefault.bird6Socket, "socket file for bird6, set either in parameter or environment variable BIRD6_SOCKET")
listenParam := flag.String("listen", settingDefault.listen, "listen address, set either in parameter or environment variable BIRDLG_LISTEN") listenParam := flag.String("listen", settingDefault.listen, "listen address, set either in parameter or environment variable BIRDLG_LISTEN")
AllowedIPsParam := flag.String("allowed", strings.Join(settingDefault.allowedIPs, ","), "IPs allowed to access this proxy, separated by commas. Don't set to allow all IPs.") AllowedIPsParam := flag.String("allowed", strings.Join(settingDefault.allowedIPs, ","), "IPs allowed to access this proxy, separated by commas. Don't set to allow all IPs.")
flag.Parse() flag.Parse()
setting.birdSocket = *birdParam setting.birdSocket = *birdParam
setting.bird6Socket = *bird6Param
setting.listen = *listenParam setting.listen = *listenParam
setting.allowedIPs = strings.Split(*AllowedIPsParam, ",") setting.allowedIPs = strings.Split(*AllowedIPsParam, ",")
// 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("/traceroute", tracerouteHandler)
http.HandleFunc("/traceroute", tracerouteIPv4Wrapper)
http.HandleFunc("/traceroute6", tracerouteIPv6Wrapper)
http.ListenAndServe(*listenParam, handlers.LoggingHandler(os.Stdout, accessHandler(http.DefaultServeMux))) http.ListenAndServe(*listenParam, handlers.LoggingHandler(os.Stdout, accessHandler(http.DefaultServeMux)))
} }

View File

@ -4,20 +4,12 @@ import (
"fmt" "fmt"
"net/http" "net/http"
"os/exec" "os/exec"
"regexp"
"runtime" "runtime"
"strconv"
"strings" "strings"
) )
// Wrapper of traceroute, IPv4
func tracerouteIPv4Wrapper(httpW http.ResponseWriter, httpR *http.Request) {
tracerouteRealHandler(false, httpW, httpR)
}
// Wrapper of traceroute, IPv6
func tracerouteIPv6Wrapper(httpW http.ResponseWriter, httpR *http.Request) {
tracerouteRealHandler(true, httpW, httpR)
}
func tracerouteTryExecute(cmd []string, args [][]string) ([]byte, string) { func tracerouteTryExecute(cmd []string, args [][]string) ([]byte, string) {
var output []byte var output []byte
var errString = "" var errString = ""
@ -35,8 +27,7 @@ func tracerouteTryExecute(cmd []string, args [][]string) ([]byte, string) {
return nil, errString return nil, errString
} }
// Real handler of traceroute requests func tracerouteHandler(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"))
query = strings.TrimSpace(query) query = strings.TrimSpace(query)
if query == "" { if query == "" {
@ -44,88 +35,28 @@ func tracerouteRealHandler(useIPv6 bool, httpW http.ResponseWriter, httpR *http.
} else { } else {
var result []byte var result []byte
var errString string var errString string
if runtime.GOOS == "freebsd" || runtime.GOOS == "netbsd" { skippedCounter := 0
if useIPv6 {
result, errString = tracerouteTryExecute( if runtime.GOOS == "freebsd" || runtime.GOOS == "netbsd" || runtime.GOOS == "openbsd" {
[]string{ result, errString = tracerouteTryExecute(
"traceroute6", []string{
"traceroute", "traceroute",
}, },
[][]string{ [][]string{
{"-q1", "-w1", query}, {"-q1", "-w1", query},
{"-q1", "-w1", query}, },
}, )
)
} else {
result, errString = tracerouteTryExecute(
[]string{
"traceroute",
"traceroute6",
},
[][]string{
{"-q1", "-w1", query},
{"-q1", "-w1", query},
},
)
}
} else if runtime.GOOS == "openbsd" {
if useIPv6 {
result, errString = tracerouteTryExecute(
[]string{
"traceroute6",
"traceroute",
},
[][]string{
{"-q1", "-w1", query},
{"-q1", "-w1", query},
},
)
} else {
result, errString = tracerouteTryExecute(
[]string{
"traceroute",
"traceroute6",
},
[][]string{
{"-A", "-q1", "-w1", query},
{"-A", "-q1", "-w1", query},
},
)
}
} else if runtime.GOOS == "linux" { } else if runtime.GOOS == "linux" {
if useIPv6 { result, errString = tracerouteTryExecute(
result, errString = tracerouteTryExecute( []string{
[]string{ "traceroute",
"traceroute", "traceroute",
"traceroute", },
"traceroute", [][]string{
"traceroute", {"-q1", "-N32", "-w1", query},
}, {"-q1", "-w1", query},
[][]string{ },
{"-6", "-q1", "-N32", "-w1", query}, )
{"-4", "-q1", "-N32", "-w1", query},
// For Busybox traceroute which doesn't support simultaneous requests
{"-6", "-q1", "-w1", query},
{"-4", "-q1", "-w1", query},
},
)
} else {
result, errString = tracerouteTryExecute(
[]string{
"traceroute",
"traceroute",
"traceroute",
"traceroute",
},
[][]string{
{"-4", "-q1", "-N32", "-w1", query},
{"-6", "-q1", "-N32", "-w1", query},
// For Busybox traceroute which doesn't support simultaneous requests
{"-4", "-q1", "-w1", query},
{"-6", "-q1", "-w1", query},
},
)
}
} else { } else {
httpW.WriteHeader(http.StatusInternalServerError) httpW.WriteHeader(http.StatusInternalServerError)
httpW.Write([]byte("traceroute not supported on this node.\n")) httpW.Write([]byte("traceroute not supported on this node.\n"))
@ -133,10 +64,18 @@ func tracerouteRealHandler(useIPv6 bool, httpW http.ResponseWriter, httpR *http.
} }
if errString != "" { if errString != "" {
httpW.WriteHeader(http.StatusInternalServerError) httpW.WriteHeader(http.StatusInternalServerError)
httpW.Write([]byte("traceroute returned error:\n\n" + errString)) httpW.Write([]byte(errString))
} }
if result != nil { if result != nil {
httpW.Write(result) errString = string(result)
errString = regexp.MustCompile(`\s*(\d*)\s*\*\n`).ReplaceAllStringFunc(errString, func(w string) string {
skippedCounter++
return ""
})
httpW.Write([]byte(strings.TrimSpace(errString)))
if skippedCounter > 0 {
httpW.Write([]byte("\n\n" + strconv.Itoa(skippedCounter) + " hops not responding."))
}
} }
} }
} }