| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | package template | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"regexp" | 
					
						
							|  |  |  | 	gotmpl "text/template" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/coredns/coredns/core/dnsserver" | 
					
						
							|  |  |  | 	"github.com/coredns/coredns/plugin" | 
					
						
							| 
									
										
										
										
											2018-02-16 03:45:25 -05:00
										 |  |  | 	"github.com/coredns/coredns/plugin/pkg/upstream" | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/mholt/caddy" | 
					
						
							|  |  |  | 	"github.com/miekg/dns" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func init() { | 
					
						
							|  |  |  | 	caddy.RegisterPlugin("template", caddy.Plugin{ | 
					
						
							|  |  |  | 		ServerType: "dns", | 
					
						
							|  |  |  | 		Action:     setupTemplate, | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func setupTemplate(c *caddy.Controller) error { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 	handler, err := templateParse(c) | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return plugin.Error("template", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 	if err := setupMetrics(c); err != nil { | 
					
						
							|  |  |  | 		return plugin.Error("template", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	dnsserver.GetConfig(c).AddPlugin(func(next plugin.Handler) plugin.Handler { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 		handler.Next = next | 
					
						
							|  |  |  | 		return handler | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | func templateParse(c *caddy.Controller) (handler Handler, err error) { | 
					
						
							|  |  |  | 	handler.Templates = make([]template, 0) | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for c.Next() { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 		if !c.NextArg() { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 			return handler, c.ArgErr() | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		class, ok := dns.StringToClass[c.Val()] | 
					
						
							|  |  |  | 		if !ok { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 			return handler, c.Errf("invalid query class %s", c.Val()) | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if !c.NextArg() { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 			return handler, c.ArgErr() | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 		qtype, ok := dns.StringToType[c.Val()] | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 		if !ok { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 			return handler, c.Errf("invalid RR class %s", c.Val()) | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 		zones := c.RemainingArgs() | 
					
						
							|  |  |  | 		if len(zones) == 0 { | 
					
						
							|  |  |  | 			zones = make([]string, len(c.ServerBlockKeys)) | 
					
						
							|  |  |  | 			copy(zones, c.ServerBlockKeys) | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 		for i, str := range zones { | 
					
						
							|  |  |  | 			zones[i] = plugin.Host(str).Normalize() | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 		handler.Zones = append(handler.Zones, zones...) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		t := template{qclass: class, qtype: qtype, zones: zones} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		t.regex = make([]*regexp.Regexp, 0) | 
					
						
							|  |  |  | 		templatePrefix := "" | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		t.answer = make([]*gotmpl.Template, 0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for c.NextBlock() { | 
					
						
							|  |  |  | 			switch c.Val() { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 			case "match": | 
					
						
							|  |  |  | 				args := c.RemainingArgs() | 
					
						
							|  |  |  | 				if len(args) == 0 { | 
					
						
							|  |  |  | 					return handler, c.ArgErr() | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				for _, regex := range args { | 
					
						
							|  |  |  | 					r, err := regexp.Compile(regex) | 
					
						
							|  |  |  | 					if err != nil { | 
					
						
							|  |  |  | 						return handler, c.Errf("could not parse regex: %s, %v", regex, err) | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					templatePrefix = templatePrefix + regex + " " | 
					
						
							|  |  |  | 					t.regex = append(t.regex, r) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 			case "answer": | 
					
						
							|  |  |  | 				args := c.RemainingArgs() | 
					
						
							|  |  |  | 				if len(args) == 0 { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 					return handler, c.ArgErr() | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				for _, answer := range args { | 
					
						
							|  |  |  | 					tmpl, err := gotmpl.New("answer").Parse(answer) | 
					
						
							|  |  |  | 					if err != nil { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 						return handler, c.Errf("could not compile template: %s, %v", c.Val(), err) | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					t.answer = append(t.answer, tmpl) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			case "additional": | 
					
						
							|  |  |  | 				args := c.RemainingArgs() | 
					
						
							|  |  |  | 				if len(args) == 0 { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 					return handler, c.ArgErr() | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				for _, additional := range args { | 
					
						
							|  |  |  | 					tmpl, err := gotmpl.New("additional").Parse(additional) | 
					
						
							|  |  |  | 					if err != nil { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 						return handler, c.Errf("could not compile template: %s, %v\n", c.Val(), err) | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					t.additional = append(t.additional, tmpl) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			case "authority": | 
					
						
							|  |  |  | 				args := c.RemainingArgs() | 
					
						
							|  |  |  | 				if len(args) == 0 { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 					return handler, c.ArgErr() | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				for _, authority := range args { | 
					
						
							|  |  |  | 					tmpl, err := gotmpl.New("authority").Parse(authority) | 
					
						
							|  |  |  | 					if err != nil { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 						return handler, c.Errf("could not compile template: %s, %v\n", c.Val(), err) | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					t.authority = append(t.authority, tmpl) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			case "rcode": | 
					
						
							|  |  |  | 				if !c.NextArg() { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 					return handler, c.ArgErr() | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				rcode, ok := dns.StringToRcode[c.Val()] | 
					
						
							|  |  |  | 				if !ok { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 					return handler, c.Errf("unknown rcode %s", c.Val()) | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				t.rcode = rcode | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 			case "fallthrough": | 
					
						
							| 
									
										
										
										
											2018-01-09 21:48:32 +00:00
										 |  |  | 				t.fall.SetZonesFromArgs(c.RemainingArgs()) | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-16 03:45:25 -05:00
										 |  |  | 			case "upstream": | 
					
						
							|  |  |  | 				args := c.RemainingArgs() | 
					
						
							|  |  |  | 				u, err := upstream.NewUpstream(args) | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return handler, err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				t.upstream = u | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 			default: | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 				return handler, c.ArgErr() | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 		if len(t.regex) == 0 { | 
					
						
							|  |  |  | 			t.regex = append(t.regex, regexp.MustCompile(".*")) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 		if len(t.answer) == 0 && len(t.additional) == 0 && t.rcode == dns.RcodeSuccess { | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 			return handler, c.Errf("no answer section for template found: %v", handler) | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 		handler.Templates = append(handler.Templates, t) | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:30:58 +01:00
										 |  |  | 	return | 
					
						
							| 
									
										
										
										
											2018-01-08 11:52:25 +01:00
										 |  |  | } |