added better error handling, ipv6 and ipv4 can now be disabled on demand

This commit is contained in:
Daniel Czerwonk 2016-12-11 16:26:57 +01:00
parent 2cb67f7b84
commit ded7381cb0
3 changed files with 65 additions and 27 deletions

View File

@ -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...)
} }
func getSessionsFromBird(ipVersion int) []*session { if *bird6Enabled {
s, err := getSessionsFromBird(6)
if err != nil {
return nil, err
}
sessions = append(sessions, s...)
}
return sessions, nil
}
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
View File

@ -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()
if err != nil {
log.Println(err)
http.Error(w, err.Error(), http.StatusInternalServerError)
}
_, err = w.Write(buf.Bytes())
if err != nil {
log.Println(err)
}
}
}
func handleMetricsRequest(w io.Writer, r *http.Request) error {
sessions, err := getSessions()
if err != nil {
return err
}
for _, s := range sessions { for _, s := range sessions {
writeLineForSession(s, w) writeForSession(s, w)
}
} }
func writeLineForSession(s *session, w io.Writer) { 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)

View File

@ -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)
} }