middleware/metrics: fix crash on startup (#318)

Make the methods that handle Metrics all use pointer receivers to fix
sync.Once not being initialized.

Finish the setup_test to test for failures. And make the check for the
address more strict and return an error when it does not have a port
number.

Add a toplevel test that starts a CoreDNS server with metrics enabled
so we catch these errors in the future.
This commit is contained in:
Miek Gieben 2016-10-04 11:05:04 +01:00 committed by GitHub
parent d914832904
commit db6c9a3f01
5 changed files with 53 additions and 16 deletions

View file

@ -13,7 +13,7 @@ import (
) )
// ServeDNS implements the Handler interface. // ServeDNS implements the Handler interface.
func (m Metrics) ServeDNS(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) { func (m *Metrics) ServeDNS(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) {
state := request.Request{W: w, Req: r} state := request.Request{W: w, Req: r}
qname := state.QName() qname := state.QName()

View file

@ -32,7 +32,7 @@ type Metrics struct {
Addr string Addr string
ln net.Listener ln net.Listener
mux *http.ServeMux mux *http.ServeMux
Once *sync.Once Once sync.Once
ZoneNames []string ZoneNames []string
} }

View file

@ -1,6 +1,7 @@
package metrics package metrics
import ( import (
"net"
"sync" "sync"
"github.com/miekg/coredns/core/dnsserver" "github.com/miekg/coredns/core/dnsserver"
@ -35,15 +36,15 @@ func setup(c *caddy.Controller) error {
return nil return nil
} }
func prometheusParse(c *caddy.Controller) (Metrics, error) { func prometheusParse(c *caddy.Controller) (*Metrics, error) {
var ( var (
met Metrics met = &Metrics{Addr: addr}
err error err error
) )
for c.Next() { for c.Next() {
if len(met.ZoneNames) > 0 { if len(met.ZoneNames) > 0 {
return Metrics{}, c.Err("metrics: can only have one metrics module per server") return met, c.Err("metrics: can only have one metrics module per server")
} }
met.ZoneNames = make([]string, len(c.ServerBlockKeys)) met.ZoneNames = make([]string, len(c.ServerBlockKeys))
copy(met.ZoneNames, c.ServerBlockKeys) copy(met.ZoneNames, c.ServerBlockKeys)
@ -56,26 +57,32 @@ func prometheusParse(c *caddy.Controller) (Metrics, error) {
case 0: case 0:
case 1: case 1:
met.Addr = args[0] met.Addr = args[0]
_, _, e := net.SplitHostPort(met.Addr)
if e != nil {
return met, e
}
default: default:
return Metrics{}, c.ArgErr() return met, c.ArgErr()
} }
for c.NextBlock() { for c.NextBlock() {
switch c.Val() { switch c.Val() {
case "address": case "address":
args = c.RemainingArgs() args = c.RemainingArgs()
if len(args) != 1 { if len(args) != 1 {
return Metrics{}, c.ArgErr() return met, c.ArgErr()
} }
met.Addr = args[0] met.Addr = args[0]
// expecting something that resembles a host-port
_, _, e := net.SplitHostPort(met.Addr)
if e != nil {
return met, e
}
default: default:
return Metrics{}, c.Errf("metrics: unknown item: %s", c.Val()) return met, c.Errf("metrics: unknown item: %s", c.Val())
} }
} }
} }
if met.Addr == "" {
met.Addr = addr
}
return met, err return met, err
} }

View file

@ -10,19 +10,32 @@ func TestPrometheus(t *testing.T) {
tests := []struct { tests := []struct {
input string input string
shouldErr bool shouldErr bool
addr string
}{ }{
{`prometheus`, false}, // oks
{`prometheus {}`, false}, // TODO(miek): should be true {`prometheus`, false, "localhost:9153"},
{`prometheus /foo`, false}, // TODO(miek): should be true {`prometheus localhost:53`, false, "localhost:53"},
{`prometheus localhost:53`, false}, // fails
{`prometheus {}`, true, ""},
{`prometheus /foo`, true, ""},
} }
for i, test := range tests { for i, test := range tests {
c := caddy.NewTestController("dns", test.input) c := caddy.NewTestController("dns", test.input)
err := setup(c) m, err := prometheusParse(c)
if test.shouldErr && err == nil { if test.shouldErr && err == nil {
t.Errorf("Test %v: Expected error but found nil", i) t.Errorf("Test %v: Expected error but found nil", i)
continue
} else if !test.shouldErr && err != nil { } else if !test.shouldErr && err != nil {
t.Errorf("Test %v: Expected no error but found error: %v", i, err) t.Errorf("Test %v: Expected no error but found error: %v", i, err)
continue
}
if test.shouldErr {
continue
}
if test.addr != m.Addr {
t.Errorf("Test %v: Expected address %s but found: %s", i, test.addr, m.Addr)
} }
} }
} }

17
test/metrics_test.go Normal file
View file

@ -0,0 +1,17 @@
package test
import "testing"
// Start test server that has metrics enabled. Then tear it down again.
func TestMetricsServer(t *testing.T) {
corefile := `.:0 {
chaos CoreDNS-001 miek@miek.nl
prometheus localhost:0
}
`
srv, err := CoreDNSServer(corefile)
if err != nil {
t.Fatalf("Could not get CoreDNS serving instance: %s", err)
}
defer srv.Stop()
}