package dnstap import ( "net/url" "os" "strings" "github.com/coredns/caddy" "github.com/coredns/coredns/core/dnsserver" "github.com/coredns/coredns/plugin" clog "github.com/coredns/coredns/plugin/pkg/log" ) var log = clog.NewWithPlugin("dnstap") func init() { plugin.Register("dnstap", setup) } func parseConfig(c *caddy.Controller) (Dnstap, error) { c.Next() // directive name d := Dnstap{} endpoint := "" args := c.RemainingArgs() if len(args) == 0 { return d, c.ArgErr() } endpoint = args[0] if strings.HasPrefix(endpoint, "tcp://") { // remote network endpoint endpointURL, err := url.Parse(endpoint) if err != nil { return d, c.ArgErr() } dio := newIO("tcp", endpointURL.Host) d = Dnstap{io: dio} } else { endpoint = strings.TrimPrefix(endpoint, "unix://") dio := newIO("unix", endpoint) d = Dnstap{io: dio} } d.IncludeRawMessage = len(args) == 2 && args[1] == "full" hostname, _ := os.Hostname() d.Identity = []byte(hostname) d.Version = []byte(caddy.AppName + "-" + caddy.AppVersion) for c.NextBlock() { switch c.Val() { case "identity": { if !c.NextArg() { return d, c.ArgErr() } d.Identity = []byte(c.Val()) } case "version": { if !c.NextArg() { return d, c.ArgErr() } d.Version = []byte(c.Val()) } } } return d, nil } func setup(c *caddy.Controller) error { dnstap, err := parseConfig(c) if err != nil { return plugin.Error("dnstap", err) } c.OnStartup(func() error { if err := dnstap.io.(*dio).connect(); err != nil { log.Errorf("No connection to dnstap endpoint: %s", err) } return nil }) c.OnRestart(func() error { dnstap.io.(*dio).close() return nil }) c.OnFinalShutdown(func() error { dnstap.io.(*dio).close() return nil }) dnsserver.GetConfig(c).AddPlugin( func(next plugin.Handler) plugin.Handler { dnstap.Next = next return dnstap }) return nil }