added better error handling, ipv6 and ipv4 can now be disabled on demand
This commit is contained in:
parent
2cb67f7b84
commit
ded7381cb0
@ -17,36 +17,45 @@ along with this program. If not, see <http://www.gnu.org/licenses/>
|
|||||||
|
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import "os/exec"
|
||||||
"log"
|
|
||||||
"os/exec"
|
|
||||||
)
|
|
||||||
|
|
||||||
func getSessions() []*session {
|
func getSessions() ([]*session, error) {
|
||||||
birdSessions := getSessionsFromBird(4)
|
sessions := make([]*session, 0)
|
||||||
bird6Sessions := getSessionsFromBird(6)
|
|
||||||
|
|
||||||
return append(birdSessions, bird6Sessions...)
|
if *birdEnabled {
|
||||||
|
s, err := getSessionsFromBird(4)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
sessions = append(sessions, s...)
|
||||||
|
}
|
||||||
|
|
||||||
|
if *bird6Enabled {
|
||||||
|
s, err := getSessionsFromBird(6)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
sessions = append(sessions, s...)
|
||||||
|
}
|
||||||
|
|
||||||
|
return sessions, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func getSessionsFromBird(ipVersion int) []*session {
|
func getSessionsFromBird(ipVersion int) ([]*session, error) {
|
||||||
client := *birdClient
|
client := *birdClient
|
||||||
|
|
||||||
if ipVersion == 6 {
|
if ipVersion == 6 {
|
||||||
client += "6"
|
client += "6"
|
||||||
}
|
}
|
||||||
|
|
||||||
output := getBirdOutput(client)
|
output, err := getBirdOutput(client)
|
||||||
return parseOutput(output, ipVersion)
|
|
||||||
}
|
|
||||||
|
|
||||||
func getBirdOutput(birdClient string) []byte {
|
|
||||||
b, err := exec.Command(birdClient, "show", "protocols", "all").Output()
|
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
b = make([]byte, 0)
|
return nil, err
|
||||||
log.Println(err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return b
|
return parseOutput(output, ipVersion), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func getBirdOutput(birdClient string) ([]byte, error) {
|
||||||
|
return exec.Command(birdClient, "show", "protocols", "all").Output()
|
||||||
}
|
}
|
||||||
|
43
main.go
43
main.go
@ -18,6 +18,8 @@ along with this program. If not, see <http://www.gnu.org/licenses/>
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
|
"bytes"
|
||||||
"flag"
|
"flag"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
@ -35,13 +37,15 @@ type session struct {
|
|||||||
uptime int
|
uptime int
|
||||||
}
|
}
|
||||||
|
|
||||||
const version string = "0.2.1"
|
const version string = "0.3"
|
||||||
|
|
||||||
var (
|
var (
|
||||||
showVersion = flag.Bool("version", false, "Print version information.")
|
showVersion = flag.Bool("version", false, "Print version information.")
|
||||||
listenAddress = flag.String("web.listen-address", ":9200", "Address on which to expose metrics and web interface.")
|
listenAddress = flag.String("web.listen-address", ":9200", "Address on which to expose metrics and web interface.")
|
||||||
metricsPath = flag.String("web.telemetry-path", "/metrics", "Path under which to expose metrics.")
|
metricsPath = flag.String("web.telemetry-path", "/metrics", "Path under which to expose metrics.")
|
||||||
birdClient = flag.String("bird.client", "birdc", "Binary to communicate with the bird routing daemon")
|
birdClient = flag.String("bird.client", "birdc", "Binary to communicate with the bird routing daemon")
|
||||||
|
birdEnabled = flag.Bool("bird.ipv4", true, "Get sessions from bird")
|
||||||
|
bird6Enabled = flag.Bool("bird.ipv6", true, "Get sessions from bird6")
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
@ -64,21 +68,46 @@ func printVersion() {
|
|||||||
|
|
||||||
func startServer() {
|
func startServer() {
|
||||||
fmt.Printf("Starting bird BGP exporter (Version: %s)\n", version)
|
fmt.Printf("Starting bird BGP exporter (Version: %s)\n", version)
|
||||||
http.HandleFunc(*metricsPath, handleMetricsRequest)
|
http.HandleFunc(*metricsPath, errorHandler(handleMetricsRequest))
|
||||||
|
|
||||||
fmt.Printf("Listening for %s on %s\n", *metricsPath, *listenAddress)
|
fmt.Printf("Listening for %s on %s\n", *metricsPath, *listenAddress)
|
||||||
log.Fatal(http.ListenAndServe(*listenAddress, nil))
|
log.Fatal(http.ListenAndServe(*listenAddress, nil))
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleMetricsRequest(w http.ResponseWriter, r *http.Request) {
|
func errorHandler(f func(io.Writer, *http.Request) error) http.HandlerFunc {
|
||||||
sessions := getSessions()
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
wr := bufio.NewWriter(&buf)
|
||||||
|
err := f(wr, r)
|
||||||
|
wr.Flush()
|
||||||
|
|
||||||
for _, s := range sessions {
|
if err != nil {
|
||||||
writeLineForSession(s, w)
|
log.Println(err)
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = w.Write(buf.Bytes())
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
log.Println(err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func writeLineForSession(s *session, w io.Writer) {
|
func handleMetricsRequest(w io.Writer, r *http.Request) error {
|
||||||
|
sessions, err := getSessions()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, s := range sessions {
|
||||||
|
writeForSession(s, w)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func writeForSession(s *session, w io.Writer) {
|
||||||
fmt.Fprintf(w, "bgp%d_session_up{name=\"%s\"} %d\n", s.ipVersion, s.name, s.established)
|
fmt.Fprintf(w, "bgp%d_session_up{name=\"%s\"} %d\n", s.ipVersion, s.name, s.established)
|
||||||
fmt.Fprintf(w, "bgp%d_session_prefix_count_import{name=\"%s\"} %d\n", s.ipVersion, s.name, s.imported)
|
fmt.Fprintf(w, "bgp%d_session_prefix_count_import{name=\"%s\"} %d\n", s.ipVersion, s.name, s.imported)
|
||||||
fmt.Fprintf(w, "bgp%d_session_prefix_count_export{name=\"%s\"} %d\n", s.ipVersion, s.name, s.exported)
|
fmt.Fprintf(w, "bgp%d_session_prefix_count_export{name=\"%s\"} %d\n", s.ipVersion, s.name, s.exported)
|
||||||
|
@ -48,7 +48,7 @@ func parseOutput(data []byte, ipVersion int) []*session {
|
|||||||
|
|
||||||
for scanner.Scan() {
|
for scanner.Scan() {
|
||||||
line := scanner.Text()
|
line := scanner.Text()
|
||||||
if session, res := parseLineForSession(line, ipVersion); res == true {
|
if session, ok := parseLineForSession(line, ipVersion); ok {
|
||||||
current = session
|
current = session
|
||||||
sessions = append(sessions, current)
|
sessions = append(sessions, current)
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user