Instead of hardcoding plugin lists in autopath/health, use interfaces. (#1306)

Switched health and autopath plugin to allow any plugins to be used instead
of a hardcoded list. I did not switch federation over since it wasn't
obvious that anything other than kubernetes could be used with it.

Fixes #1291
This commit is contained in:
James Hartig 2017-12-12 15:40:30 -05:00 committed by Miek Gieben
parent 99e163c375
commit a469a17cdf
8 changed files with 63 additions and 35 deletions

View file

@ -167,6 +167,21 @@ func (c *Config) Handler(name string) plugin.Handler {
return nil return nil
} }
// Handlers returns a slice of plugins that have been registered. This can be used to
// inspect and interact with registered plugins but cannot be used to remove or add plugins.
// Note that this is order dependent and the order is defined in directives.go, i.e. if your plugin
// comes before the plugin you are checking; it will not be there (yet).
func (c *Config) Handlers() []plugin.Handler {
if c.registry == nil {
return nil
}
hs := make([]plugin.Handler, 0, len(c.registry))
for k := range c.registry {
hs = append(hs, c.registry[k])
}
return hs
}
// groupSiteConfigsByListenAddr groups site configs by their listen // groupSiteConfigsByListenAddr groups site configs by their listen
// (bind) address, so sites that use the same listener can be served // (bind) address, so sites that use the same listener can be served
// on the same server instance. The return value maps the listen // on the same server instance. The return value maps the listen

View file

@ -0,0 +1,31 @@
package dnsserver
import (
"testing"
)
func TestHandler(t *testing.T) {
tp := testPlugin{}
c := testConfig("dns", tp)
if _, err := NewServer("127.0.0.1:53", []*Config{c}); err != nil {
t.Errorf("Expected no error for NewServer, got %s", err)
}
if h := c.Handler("testplugin"); h != tp {
t.Errorf("Expected testPlugin from Handler, got %T", h)
}
if h := c.Handler("nothing"); h != nil {
t.Errorf("Expected nil from Handler, got %T", h)
}
}
func TestHandlers(t *testing.T) {
tp := testPlugin{}
c := testConfig("dns", tp)
if _, err := NewServer("127.0.0.1:53", []*Config{c}); err != nil {
t.Errorf("Expected no error for NewServer, got %s", err)
}
hs := c.Handlers()
if len(hs) != 1 || hs[0] != tp {
t.Errorf("Expected [testPlugin] from Handlers, got %v", hs)
}
}

View file

@ -19,10 +19,7 @@ autopath [ZONE...] RESOLV-CONF
plugin. For instance `@kubernetes`, will call out to the kubernetes plugin (for each plugin. For instance `@kubernetes`, will call out to the kubernetes plugin (for each
query) to retrieve the search list it should use. query) to retrieve the search list it should use.
Currently the following set of plugin has implemented *autopath*: If a plugin implements the `AutoPather` interface then it can be used.
* *kubernetes*
* *erratic*
## Metrics ## Metrics

View file

@ -46,6 +46,12 @@ import (
// If Func returns a nil slice, no autopathing will be done. // If Func returns a nil slice, no autopathing will be done.
type Func func(request.Request) []string type Func func(request.Request) []string
// AutoPather defines the interface that a plugin should implement in order to be
// used by AutoPath.
type AutoPather interface {
AutoPath(request.Request) []string
}
// AutoPath perform autopath: service side search path completion. // AutoPath perform autopath: service side search path completion.
type AutoPath struct { type AutoPath struct {
Next plugin.Handler Next plugin.Handler

View file

@ -5,8 +5,6 @@ import (
"github.com/coredns/coredns/core/dnsserver" "github.com/coredns/coredns/core/dnsserver"
"github.com/coredns/coredns/plugin" "github.com/coredns/coredns/plugin"
"github.com/coredns/coredns/plugin/erratic"
"github.com/coredns/coredns/plugin/kubernetes"
"github.com/mholt/caddy" "github.com/mholt/caddy"
"github.com/miekg/dns" "github.com/miekg/dns"
@ -34,11 +32,10 @@ func setup(c *caddy.Controller) error {
if m == nil { if m == nil {
return nil return nil
} }
if x, ok := m.(*kubernetes.Kubernetes); ok { if x, ok := m.(AutoPather); ok {
ap.searchFunc = x.AutoPath
}
if x, ok := m.(*erratic.Erratic); ok {
ap.searchFunc = x.AutoPath ap.searchFunc = x.AutoPath
} else {
return plugin.Error("autopath", fmt.Errorf("%s does not implement the AutoPather interface", mw))
} }
return nil return nil
}) })
@ -51,12 +48,6 @@ func setup(c *caddy.Controller) error {
return nil return nil
} }
// allowedPlugins has a list of plugin that can be used by autopath.
var allowedPlugins = map[string]bool{
"@kubernetes": true,
"@erratic": true,
}
func autoPathParse(c *caddy.Controller) (*AutoPath, string, error) { func autoPathParse(c *caddy.Controller) (*AutoPath, string, error) {
ap := &AutoPath{} ap := &AutoPath{}
mw := "" mw := ""
@ -68,10 +59,7 @@ func autoPathParse(c *caddy.Controller) (*AutoPath, string, error) {
} }
resolv := zoneAndresolv[len(zoneAndresolv)-1] resolv := zoneAndresolv[len(zoneAndresolv)-1]
if resolv[0] == '@' { if resolv[0] == '@' {
_, ok := allowedPlugins[resolv] mw = resolv[1:]
if ok {
mw = resolv[1:]
}
} else { } else {
// assume file on disk // assume file on disk
rc, err := dns.ClientConfigFromFile(resolv) rc, err := dns.ClientConfigFromFile(resolv)

View file

@ -18,10 +18,7 @@ supports health checks has a section "Health" in their README.
## Plugins ## Plugins
The following plugins report health to the health plugin: Any plugin that implements the Healther interface will be used to report health.
* erratic
* kubernetes
## Examples ## Examples

View file

@ -36,9 +36,3 @@ func (h *health) poll() {
} }
h.SetOk(true) h.SetOk(true)
} }
// Plugins that implements the Healther interface.
var healthers = map[string]bool{
"erratic": true,
"kubernetes": true,
}

View file

@ -26,9 +26,9 @@ func setup(c *caddy.Controller) error {
h := &health{Addr: addr} h := &health{Addr: addr}
c.OnStartup(func() error { c.OnStartup(func() error {
for he := range healthers { plugins := dnsserver.GetConfig(c).Handlers()
m := dnsserver.GetConfig(c).Handler(he) for _, p := range plugins {
if x, ok := m.(Healther); ok { if x, ok := p.(Healther); ok {
h.h = append(h.h, x) h.h = append(h.h, x)
} }
} }