| 
									
										
										
										
											2016-09-07 11:10:16 +01:00
										 |  |  | package replacer
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | import (
 | 
					
						
							| 
									
										
										
										
											2018-11-13 14:20:49 -05:00
										 |  |  | 	"context"
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | 	"strconv"
 | 
					
						
							|  |  |  | 	"strings"
 | 
					
						
							|  |  |  | 	"time"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-13 14:20:49 -05:00
										 |  |  | 	"github.com/coredns/coredns/plugin/metadata"
 | 
					
						
							| 
									
										
										
										
											2017-09-21 15:15:47 +01:00
										 |  |  | 	"github.com/coredns/coredns/plugin/pkg/dnstest"
 | 
					
						
							| 
									
										
										
										
											2017-02-21 22:51:47 -08:00
										 |  |  | 	"github.com/coredns/coredns/request"
 | 
					
						
							| 
									
										
										
										
											2016-09-07 11:10:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | 	"github.com/miekg/dns"
 | 
					
						
							|  |  |  | )
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | // Replacer replaces labels for values in strings.
 | 
					
						
							|  |  |  | type Replacer struct {
 | 
					
						
							|  |  |  | 	valueFunc func(request.Request, *dnstest.Recorder, string) string
 | 
					
						
							|  |  |  | 	labels    []string
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | // labels are all supported labels that can be used in the default Replacer.
 | 
					
						
							|  |  |  | var labels = []string{
 | 
					
						
							|  |  |  | 	"{type}",
 | 
					
						
							|  |  |  | 	"{name}",
 | 
					
						
							|  |  |  | 	"{class}",
 | 
					
						
							|  |  |  | 	"{proto}",
 | 
					
						
							|  |  |  | 	"{size}",
 | 
					
						
							|  |  |  | 	"{remote}",
 | 
					
						
							|  |  |  | 	"{port}",
 | 
					
						
							|  |  |  | 	"{local}",
 | 
					
						
							|  |  |  | 	// Header values.
 | 
					
						
							|  |  |  | 	headerReplacer + "id}",
 | 
					
						
							|  |  |  | 	headerReplacer + "opcode}",
 | 
					
						
							|  |  |  | 	headerReplacer + "do}",
 | 
					
						
							|  |  |  | 	headerReplacer + "bufsize}",
 | 
					
						
							|  |  |  | 	// Recorded replacements.
 | 
					
						
							|  |  |  | 	"{rcode}",
 | 
					
						
							|  |  |  | 	"{rsize}",
 | 
					
						
							|  |  |  | 	"{duration}",
 | 
					
						
							| 
									
										
										
										
											2019-02-21 07:13:05 +00:00
										 |  |  | 	headerReplacer + "rflags}",
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | // value returns the current value of label.
 | 
					
						
							|  |  |  | func value(state request.Request, rr *dnstest.Recorder, label string) string {
 | 
					
						
							|  |  |  | 	switch label {
 | 
					
						
							|  |  |  | 	case "{type}":
 | 
					
						
							|  |  |  | 		return state.Type()
 | 
					
						
							|  |  |  | 	case "{name}":
 | 
					
						
							|  |  |  | 		return state.Name()
 | 
					
						
							|  |  |  | 	case "{class}":
 | 
					
						
							|  |  |  | 		return state.Class()
 | 
					
						
							|  |  |  | 	case "{proto}":
 | 
					
						
							|  |  |  | 		return state.Proto()
 | 
					
						
							|  |  |  | 	case "{size}":
 | 
					
						
							|  |  |  | 		return strconv.Itoa(state.Req.Len())
 | 
					
						
							|  |  |  | 	case "{remote}":
 | 
					
						
							|  |  |  | 		return addrToRFC3986(state.IP())
 | 
					
						
							|  |  |  | 	case "{port}":
 | 
					
						
							|  |  |  | 		return state.Port()
 | 
					
						
							|  |  |  | 	case "{local}":
 | 
					
						
							|  |  |  | 		return addrToRFC3986(state.LocalIP())
 | 
					
						
							|  |  |  | 	// Header placeholders (case-insensitive).
 | 
					
						
							|  |  |  | 	case headerReplacer + "id}":
 | 
					
						
							|  |  |  | 		return strconv.Itoa(int(state.Req.Id))
 | 
					
						
							|  |  |  | 	case headerReplacer + "opcode}":
 | 
					
						
							|  |  |  | 		return strconv.Itoa(state.Req.Opcode)
 | 
					
						
							|  |  |  | 	case headerReplacer + "do}":
 | 
					
						
							|  |  |  | 		return boolToString(state.Do())
 | 
					
						
							|  |  |  | 	case headerReplacer + "bufsize}":
 | 
					
						
							|  |  |  | 		return strconv.Itoa(state.Size())
 | 
					
						
							|  |  |  | 	// Recorded replacements.
 | 
					
						
							|  |  |  | 	case "{rcode}":
 | 
					
						
							|  |  |  | 		if rr == nil {
 | 
					
						
							|  |  |  | 			return EmptyValue
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							| 
									
										
										
										
											2016-09-07 11:10:16 +01:00
										 |  |  | 		rcode := dns.RcodeToString[rr.Rcode]
 | 
					
						
							| 
									
										
										
										
											2016-03-19 20:17:44 +00:00
										 |  |  | 		if rcode == "" {
 | 
					
						
							| 
									
										
										
										
											2016-09-07 11:10:16 +01:00
										 |  |  | 			rcode = strconv.Itoa(rr.Rcode)
 | 
					
						
							| 
									
										
										
										
											2016-03-19 20:17:44 +00:00
										 |  |  | 		}
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | 		return rcode
 | 
					
						
							|  |  |  | 	case "{rsize}":
 | 
					
						
							|  |  |  | 		if rr == nil {
 | 
					
						
							|  |  |  | 			return EmptyValue
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 		return strconv.Itoa(rr.Len)
 | 
					
						
							|  |  |  | 	case "{duration}":
 | 
					
						
							|  |  |  | 		if rr == nil {
 | 
					
						
							|  |  |  | 			return EmptyValue
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 		return strconv.FormatFloat(time.Since(rr.Start).Seconds(), 'f', -1, 64) + "s"
 | 
					
						
							| 
									
										
										
										
											2019-02-21 07:13:05 +00:00
										 |  |  | 	case headerReplacer + "rflags}":
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | 		if rr != nil && rr.Msg != nil {
 | 
					
						
							|  |  |  | 			return flagsToString(rr.Msg.MsgHdr)
 | 
					
						
							| 
									
										
										
										
											2017-08-07 03:49:40 -07:00
										 |  |  | 		}
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | 		return EmptyValue
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | 	}
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | 	return EmptyValue
 | 
					
						
							|  |  |  | }
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | // New makes a new replacer. This only needs to be called once in the setup and then call Replace for each incoming message.
 | 
					
						
							|  |  |  | // A replacer is safe for concurrent use.
 | 
					
						
							|  |  |  | func New() Replacer {
 | 
					
						
							|  |  |  | 	return Replacer{
 | 
					
						
							|  |  |  | 		valueFunc: value,
 | 
					
						
							|  |  |  | 		labels:    labels,
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | // Replace performs a replacement of values on s and returns the string with the replaced values.
 | 
					
						
							|  |  |  | func (r Replacer) Replace(ctx context.Context, state request.Request, rr *dnstest.Recorder, s string) string {
 | 
					
						
							|  |  |  | 	for _, placeholder := range r.labels {
 | 
					
						
							|  |  |  | 		if strings.Contains(s, placeholder) {
 | 
					
						
							|  |  |  | 			s = strings.Replace(s, placeholder, r.valueFunc(state, rr, placeholder), -1)
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | 	// Metadata label replacements. Scan for {/ and search for next }, replace that metadata label with
 | 
					
						
							|  |  |  | 	// any meta data that is available.
 | 
					
						
							|  |  |  | 	b := strings.Builder{}
 | 
					
						
							|  |  |  | 	for strings.Contains(s, labelReplacer) {
 | 
					
						
							|  |  |  | 		idxStart := strings.Index(s, labelReplacer)
 | 
					
						
							|  |  |  | 		endOffset := idxStart + len(labelReplacer)
 | 
					
						
							|  |  |  | 		idxEnd := strings.Index(s[endOffset:], "}")
 | 
					
						
							|  |  |  | 		if idxEnd > -1 {
 | 
					
						
							|  |  |  | 			label := s[idxStart+2 : endOffset+idxEnd]
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			fm := metadata.ValueFunc(ctx, label)
 | 
					
						
							|  |  |  | 			replacement := EmptyValue
 | 
					
						
							|  |  |  | 			if fm != nil {
 | 
					
						
							|  |  |  | 				replacement = fm()
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							| 
									
										
										
										
											2018-11-13 14:20:49 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | 			b.WriteString(s[:idxStart])
 | 
					
						
							|  |  |  | 			b.WriteString(replacement)
 | 
					
						
							|  |  |  | 			s = s[endOffset+idxEnd+1:]
 | 
					
						
							|  |  |  | 		} else {
 | 
					
						
							|  |  |  | 			break
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | 	b.WriteString(s)
 | 
					
						
							|  |  |  | 	return b.String()
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-03 17:16:46 +01:00
										 |  |  | func boolToString(b bool) string {
 | 
					
						
							|  |  |  | 	if b {
 | 
					
						
							|  |  |  | 		return "true"
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	return "false"
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-07 03:49:40 -07:00
										 |  |  | // flagsToString checks all header flags and returns those
 | 
					
						
							|  |  |  | // that are set as a string separated with commas
 | 
					
						
							|  |  |  | func flagsToString(h dns.MsgHdr) string {
 | 
					
						
							|  |  |  | 	flags := make([]string, 7)
 | 
					
						
							|  |  |  | 	i := 0
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if h.Response {
 | 
					
						
							|  |  |  | 		flags[i] = "qr"
 | 
					
						
							|  |  |  | 		i++
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if h.Authoritative {
 | 
					
						
							|  |  |  | 		flags[i] = "aa"
 | 
					
						
							|  |  |  | 		i++
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	if h.Truncated {
 | 
					
						
							|  |  |  | 		flags[i] = "tc"
 | 
					
						
							|  |  |  | 		i++
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	if h.RecursionDesired {
 | 
					
						
							|  |  |  | 		flags[i] = "rd"
 | 
					
						
							|  |  |  | 		i++
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	if h.RecursionAvailable {
 | 
					
						
							|  |  |  | 		flags[i] = "ra"
 | 
					
						
							|  |  |  | 		i++
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	if h.Zero {
 | 
					
						
							|  |  |  | 		flags[i] = "z"
 | 
					
						
							|  |  |  | 		i++
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	if h.AuthenticatedData {
 | 
					
						
							|  |  |  | 		flags[i] = "ad"
 | 
					
						
							|  |  |  | 		i++
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	if h.CheckingDisabled {
 | 
					
						
							|  |  |  | 		flags[i] = "cd"
 | 
					
						
							|  |  |  | 		i++
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	return strings.Join(flags[:i], ",")
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 18:15:12 -08:00
										 |  |  | // addrToRFC3986 will add brackets to the address if it is an IPv6 address.
 | 
					
						
							|  |  |  | func addrToRFC3986(addr string) string {
 | 
					
						
							|  |  |  | 	if strings.Contains(addr, ":") {
 | 
					
						
							|  |  |  | 		return "[" + addr + "]"
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	return addr
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-13 14:20:49 -05:00
										 |  |  | const (
 | 
					
						
							| 
									
										
										
										
											2019-02-12 07:38:49 +00:00
										 |  |  | 	headerReplacer = "{>"
 | 
					
						
							|  |  |  | 	labelReplacer  = "{/"
 | 
					
						
							|  |  |  | 	// EmptyValue is the default empty value.
 | 
					
						
							|  |  |  | 	EmptyValue = "-"
 | 
					
						
							| 
									
										
										
										
											2018-11-13 14:20:49 -05:00
										 |  |  | )
 |