removed bird client dependency, using socket instead

This commit is contained in:
Daniel Czerwonk 2017-06-19 21:04:09 +02:00
parent a82100a3ed
commit 900ff9f4c2
2 changed files with 63 additions and 12 deletions

View File

@ -1,16 +1,30 @@
package main package main
import ( import (
"os/exec" "net"
"regexp"
"time"
"github.com/czerwonk/bird_exporter/protocol" "github.com/czerwonk/bird_exporter/protocol"
"github.com/prometheus/common/log"
) )
const bufferSize = 4096
const timeout = 30
var birdReturnCodeRegex *regexp.Regexp
func init() {
birdReturnCodeRegex = regexp.MustCompile("\\d{4} \n$")
}
func getProtocols() ([]*protocol.Protocol, error) { func getProtocols() ([]*protocol.Protocol, error) {
protocols := make([]*protocol.Protocol, 0) protocols := make([]*protocol.Protocol, 0)
if *birdEnabled { if *birdEnabled {
s, err := getProtocolsFromBird(4) s, err := getProtocolsFromBird(*birdSocket, 4)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -18,7 +32,7 @@ func getProtocols() ([]*protocol.Protocol, error) {
} }
if *bird6Enabled { if *bird6Enabled {
s, err := getProtocolsFromBird(6) s, err := getProtocolsFromBird(*bird6Socket, 6)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -28,14 +42,28 @@ func getProtocols() ([]*protocol.Protocol, error) {
return protocols, nil return protocols, nil
} }
func getProtocolsFromBird(ipVersion int) ([]*protocol.Protocol, error) { func getProtocolsFromBird(socketPath string, ipVersion int) ([]*protocol.Protocol, error) {
client := *birdClient conn, err := net.Dial("unix", socketPath)
if err != nil {
return nil, err
}
defer conn.Close()
if ipVersion == 6 { conn.SetReadDeadline(time.Now().Add(timeout * time.Second))
client += "6"
buf := make([]byte, bufferSize)
n, err := conn.Read(buf[:])
if err != nil {
return nil, err
}
log.Debug(string(buf[:n]))
_, err = conn.Write([]byte("show protocols all\n"))
if err != nil {
return nil, err
} }
output, err := getBirdOutput(client) output, err := readFromSocket(conn)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -43,6 +71,28 @@ func getProtocolsFromBird(ipVersion int) ([]*protocol.Protocol, error) {
return parseOutput(output, ipVersion), nil return parseOutput(output, ipVersion), nil
} }
func getBirdOutput(birdClient string) ([]byte, error) { func readFromSocket(conn net.Conn) ([]byte, error) {
return exec.Command(birdClient, "show", "protocols", "all").Output() b := make([]byte, 0)
buf := make([]byte, bufferSize)
done := false
for !done {
n, err := conn.Read(buf[:])
if err != nil {
return nil, err
}
b = append(b, buf[:n]...)
done = endsWithBirdReturnCode(b)
}
return b, nil
}
func endsWithBirdReturnCode(b []byte) bool {
if len(b) < 6 {
return false
}
return birdReturnCodeRegex.Match(b[len(b)-6:])
} }

View File

@ -11,13 +11,14 @@ import (
"github.com/prometheus/common/log" "github.com/prometheus/common/log"
) )
const version string = "0.7.1" const version string = "0.8.0"
var ( var (
showVersion = flag.Bool("version", false, "Print version information.") showVersion = flag.Bool("version", false, "Print version information.")
listenAddress = flag.String("web.listen-address", ":9324", "Address on which to expose metrics and web interface.") listenAddress = flag.String("web.listen-address", ":9324", "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") birdSocket = flag.String("bird.socket", "/var/run/bird.ctl", "Socket to communicate with bird routing daemon")
bird6Socket = flag.String("bird.socket6", "/var/run/bird6.ctl", "Socket to communicate with bird6 routing daemon")
birdEnabled = flag.Bool("bird.ipv4", true, "Get protocols from bird") birdEnabled = flag.Bool("bird.ipv4", true, "Get protocols from bird")
bird6Enabled = flag.Bool("bird.ipv6", true, "Get protocols from bird6") bird6Enabled = flag.Bool("bird.ipv6", true, "Get protocols from bird6")
) )