mirror of
				https://github.com/coredns/coredns.git
				synced 2025-10-31 02:03:20 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			196 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			196 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package replacer
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/coredns/coredns/plugin/metadata"
 | |
| 	"github.com/coredns/coredns/plugin/pkg/dnstest"
 | |
| 	"github.com/coredns/coredns/request"
 | |
| 
 | |
| 	"github.com/miekg/dns"
 | |
| )
 | |
| 
 | |
| // Replacer is a type which can replace placeholder
 | |
| // substrings in a string with actual values from a
 | |
| // dns.Msg and responseRecorder. Always use
 | |
| // NewReplacer to get one of these.
 | |
| type Replacer interface {
 | |
| 	Replace(string) string
 | |
| 	Set(key, value string)
 | |
| }
 | |
| 
 | |
| type replacer struct {
 | |
| 	ctx          context.Context
 | |
| 	replacements map[string]string
 | |
| 	emptyValue   string
 | |
| }
 | |
| 
 | |
| // New makes a new replacer based on r and rr.
 | |
| // Do not create a new replacer until r and rr have all
 | |
| // the needed values, because this function copies those
 | |
| // values into the replacer. rr may be nil if it is not
 | |
| // available. emptyValue should be the string that is used
 | |
| // in place of empty string (can still be empty string).
 | |
| func New(ctx context.Context, r *dns.Msg, rr *dnstest.Recorder, emptyValue string) Replacer {
 | |
| 	req := request.Request{W: rr, Req: r}
 | |
| 	rep := replacer{
 | |
| 		ctx: ctx,
 | |
| 		replacements: map[string]string{
 | |
| 			"{type}":   req.Type(),
 | |
| 			"{name}":   req.Name(),
 | |
| 			"{class}":  req.Class(),
 | |
| 			"{proto}":  req.Proto(),
 | |
| 			"{when}":   "", // made a noop
 | |
| 			"{size}":   strconv.Itoa(req.Len()),
 | |
| 			"{remote}": addrToRFC3986(req.IP()),
 | |
| 			"{port}":   req.Port(),
 | |
| 			"{local}":  addrToRFC3986(req.LocalIP()),
 | |
| 		},
 | |
| 		emptyValue: emptyValue,
 | |
| 	}
 | |
| 	if rr != nil {
 | |
| 		rcode := dns.RcodeToString[rr.Rcode]
 | |
| 		if rcode == "" {
 | |
| 			rcode = strconv.Itoa(rr.Rcode)
 | |
| 		}
 | |
| 		rep.replacements["{rcode}"] = rcode
 | |
| 		rep.replacements["{rsize}"] = strconv.Itoa(rr.Len)
 | |
| 		rep.replacements["{duration}"] = strconv.FormatFloat(time.Since(rr.Start).Seconds(), 'f', -1, 64) + "s"
 | |
| 		if rr.Msg != nil {
 | |
| 			rep.replacements[headerReplacer+"rflags}"] = flagsToString(rr.Msg.MsgHdr)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Header placeholders (case-insensitive)
 | |
| 	rep.replacements[headerReplacer+"id}"] = strconv.Itoa(int(r.Id))
 | |
| 	rep.replacements[headerReplacer+"opcode}"] = strconv.Itoa(r.Opcode)
 | |
| 	rep.replacements[headerReplacer+"do}"] = boolToString(req.Do())
 | |
| 	rep.replacements[headerReplacer+"bufsize}"] = strconv.Itoa(req.Size())
 | |
| 
 | |
| 	return rep
 | |
| }
 | |
| 
 | |
| // Replace performs a replacement of values on s and returns
 | |
| // the string with the replaced values.
 | |
| func (r replacer) Replace(s string) string {
 | |
| 
 | |
| 	// declare a function that replace based on header matching
 | |
| 	fscanAndReplace := func(s string, header string, replace func(string) string) string {
 | |
| 		b := strings.Builder{}
 | |
| 		for strings.Contains(s, header) {
 | |
| 			idxStart := strings.Index(s, header)
 | |
| 			endOffset := idxStart + len(header)
 | |
| 			idxEnd := strings.Index(s[endOffset:], "}")
 | |
| 			if idxEnd > -1 {
 | |
| 				placeholder := strings.ToLower(s[idxStart : endOffset+idxEnd+1])
 | |
| 				replacement := replace(placeholder)
 | |
| 				if replacement == "" {
 | |
| 					replacement = r.emptyValue
 | |
| 				}
 | |
| 				b.WriteString(s[:idxStart])
 | |
| 				b.WriteString(replacement)
 | |
| 				s = s[endOffset+idxEnd+1:]
 | |
| 			} else {
 | |
| 				break
 | |
| 			}
 | |
| 		}
 | |
| 		b.WriteString(s)
 | |
| 		return b.String()
 | |
| 	}
 | |
| 
 | |
| 	// Header replacements - these are case-insensitive, so we can't just use strings.Replace()
 | |
| 	s = fscanAndReplace(s, headerReplacer, func(placeholder string) string {
 | |
| 		return r.replacements[placeholder]
 | |
| 	})
 | |
| 
 | |
| 	// Regular replacements - these are easier because they're case-sensitive
 | |
| 	for placeholder, replacement := range r.replacements {
 | |
| 		if replacement == "" {
 | |
| 			replacement = r.emptyValue
 | |
| 		}
 | |
| 		s = strings.Replace(s, placeholder, replacement, -1)
 | |
| 	}
 | |
| 
 | |
| 	// Metadata label replacements
 | |
| 	s = fscanAndReplace(s, headerLabelReplacer, func(placeholder string) string {
 | |
| 		// label place holder has the format {/<label>}
 | |
| 		fm := metadata.ValueFunc(r.ctx, placeholder[len(headerLabelReplacer):len(placeholder)-1])
 | |
| 		if fm != nil {
 | |
| 			return fm()
 | |
| 		}
 | |
| 		return ""
 | |
| 	})
 | |
| 
 | |
| 	return s
 | |
| }
 | |
| 
 | |
| // Set sets key to value in the replacements map.
 | |
| func (r replacer) Set(key, value string) {
 | |
| 	r.replacements["{"+key+"}"] = value
 | |
| }
 | |
| 
 | |
| func boolToString(b bool) string {
 | |
| 	if b {
 | |
| 		return "true"
 | |
| 	}
 | |
| 	return "false"
 | |
| }
 | |
| 
 | |
| // 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], ",")
 | |
| }
 | |
| 
 | |
| // 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
 | |
| }
 | |
| 
 | |
| const (
 | |
| 	headerReplacer      = "{>"
 | |
| 	headerLabelReplacer = "{/"
 | |
| )
 |