| 
									
										
										
										
											2017-07-24 23:12:50 +02:00
										 |  |  | package dnstap | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2017-09-01 14:07:21 +02:00
										 |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2017-07-24 23:12:50 +02:00
										 |  |  | 	"log" | 
					
						
							| 
									
										
										
										
											2017-09-01 14:07:21 +02:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2017-07-24 23:12:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/coredns/coredns/core/dnsserver" | 
					
						
							|  |  |  | 	"github.com/coredns/coredns/middleware" | 
					
						
							|  |  |  | 	"github.com/coredns/coredns/middleware/dnstap/out" | 
					
						
							| 
									
										
										
										
											2017-09-01 14:07:21 +02:00
										 |  |  | 	"github.com/coredns/coredns/middleware/pkg/dnsutil" | 
					
						
							| 
									
										
										
										
											2017-07-24 23:12:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/mholt/caddy" | 
					
						
							|  |  |  | 	"github.com/mholt/caddy/caddyfile" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func init() { | 
					
						
							|  |  |  | 	caddy.RegisterPlugin("dnstap", caddy.Plugin{ | 
					
						
							|  |  |  | 		ServerType: "dns", | 
					
						
							|  |  |  | 		Action:     wrapSetup, | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func wrapSetup(c *caddy.Controller) error { | 
					
						
							|  |  |  | 	if err := setup(c); err != nil { | 
					
						
							|  |  |  | 		return middleware.Error("dnstap", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-01 14:07:21 +02:00
										 |  |  | type config struct { | 
					
						
							|  |  |  | 	target string | 
					
						
							|  |  |  | 	socket bool | 
					
						
							|  |  |  | 	full   bool | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func parseConfig(d *caddyfile.Dispenser) (c config, err error) { | 
					
						
							|  |  |  | 	d.Next() // directive name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !d.Args(&c.target) { | 
					
						
							|  |  |  | 		return c, d.ArgErr() | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-07-24 23:12:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-01 14:07:21 +02:00
										 |  |  | 	if strings.HasPrefix(c.target, "tcp://") { | 
					
						
							|  |  |  | 		// remote IP endpoint | 
					
						
							|  |  |  | 		servers, err := dnsutil.ParseHostPortOrFile(c.target[6:]) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return c, d.ArgErr() | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		c.target = servers[0] | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		// default to UNIX socket | 
					
						
							|  |  |  | 		if strings.HasPrefix(c.target, "unix://") { | 
					
						
							|  |  |  | 			c.target = c.target[7:] | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		c.socket = true | 
					
						
							| 
									
										
										
										
											2017-07-24 23:12:50 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-01 14:07:21 +02:00
										 |  |  | 	c.full = d.NextArg() && d.Val() == "full" | 
					
						
							| 
									
										
										
										
											2017-07-24 23:12:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func setup(c *caddy.Controller) error { | 
					
						
							| 
									
										
										
										
											2017-09-01 14:07:21 +02:00
										 |  |  | 	conf, err := parseConfig(&c.Dispenser) | 
					
						
							| 
									
										
										
										
											2017-07-24 23:12:50 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-01 14:07:21 +02:00
										 |  |  | 	dnstap := Dnstap{Pack: conf.full} | 
					
						
							| 
									
										
										
										
											2017-07-24 23:12:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-01 14:07:21 +02:00
										 |  |  | 	var o io.WriteCloser | 
					
						
							|  |  |  | 	if conf.socket { | 
					
						
							|  |  |  | 		o, err = out.NewSocket(conf.target) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			log.Printf("[WARN] Can't connect to %s at the moment: %s", conf.target, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		o = out.NewTCP(conf.target) | 
					
						
							| 
									
										
										
										
											2017-07-24 23:12:50 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	dnstap.Out = o | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c.OnShutdown(func() error { | 
					
						
							|  |  |  | 		if err := o.Close(); err != nil { | 
					
						
							|  |  |  | 			return fmt.Errorf("output: %s", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dnsserver.GetConfig(c).AddMiddleware( | 
					
						
							|  |  |  | 		func(next middleware.Handler) middleware.Handler { | 
					
						
							|  |  |  | 			dnstap.Next = next | 
					
						
							|  |  |  | 			return dnstap | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } |