mirror of
				https://github.com/coredns/coredns.git
				synced 2025-11-03 18:53:13 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			295 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			295 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package rewrite
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"fmt"
 | 
						|
	"regexp"
 | 
						|
	"strconv"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	"github.com/coredns/coredns/plugin"
 | 
						|
	"github.com/coredns/coredns/request"
 | 
						|
)
 | 
						|
 | 
						|
type exactNameRule struct {
 | 
						|
	NextAction string
 | 
						|
	From       string
 | 
						|
	To         string
 | 
						|
	ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
type prefixNameRule struct {
 | 
						|
	NextAction  string
 | 
						|
	Prefix      string
 | 
						|
	Replacement string
 | 
						|
}
 | 
						|
 | 
						|
type suffixNameRule struct {
 | 
						|
	NextAction  string
 | 
						|
	Suffix      string
 | 
						|
	Replacement string
 | 
						|
}
 | 
						|
 | 
						|
type substringNameRule struct {
 | 
						|
	NextAction  string
 | 
						|
	Substring   string
 | 
						|
	Replacement string
 | 
						|
}
 | 
						|
 | 
						|
type regexNameRule struct {
 | 
						|
	NextAction  string
 | 
						|
	Pattern     *regexp.Regexp
 | 
						|
	Replacement string
 | 
						|
	ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
const (
 | 
						|
	// ExactMatch matches only on exact match of the name in the question section of a request
 | 
						|
	ExactMatch = "exact"
 | 
						|
	// PrefixMatch matches when the name begins with the matching string
 | 
						|
	PrefixMatch = "prefix"
 | 
						|
	// SuffixMatch matches when the name ends with the matching string
 | 
						|
	SuffixMatch = "suffix"
 | 
						|
	// SubstringMatch matches on partial match of the name in the question section of a request
 | 
						|
	SubstringMatch = "substring"
 | 
						|
	// RegexMatch matches when the name in the question section of a request matches a regular expression
 | 
						|
	RegexMatch = "regex"
 | 
						|
)
 | 
						|
 | 
						|
// Rewrite rewrites the current request based upon exact match of the name
 | 
						|
// in the question section of the request.
 | 
						|
func (rule *exactNameRule) Rewrite(ctx context.Context, state request.Request) Result {
 | 
						|
	if rule.From == state.Name() {
 | 
						|
		state.Req.Question[0].Name = rule.To
 | 
						|
		return RewriteDone
 | 
						|
	}
 | 
						|
	return RewriteIgnored
 | 
						|
}
 | 
						|
 | 
						|
// Rewrite rewrites the current request when the name begins with the matching string.
 | 
						|
func (rule *prefixNameRule) Rewrite(ctx context.Context, state request.Request) Result {
 | 
						|
	if strings.HasPrefix(state.Name(), rule.Prefix) {
 | 
						|
		state.Req.Question[0].Name = rule.Replacement + strings.TrimPrefix(state.Name(), rule.Prefix)
 | 
						|
		return RewriteDone
 | 
						|
	}
 | 
						|
	return RewriteIgnored
 | 
						|
}
 | 
						|
 | 
						|
// Rewrite rewrites the current request when the name ends with the matching string.
 | 
						|
func (rule *suffixNameRule) Rewrite(ctx context.Context, state request.Request) Result {
 | 
						|
	if strings.HasSuffix(state.Name(), rule.Suffix) {
 | 
						|
		state.Req.Question[0].Name = strings.TrimSuffix(state.Name(), rule.Suffix) + rule.Replacement
 | 
						|
		return RewriteDone
 | 
						|
	}
 | 
						|
	return RewriteIgnored
 | 
						|
}
 | 
						|
 | 
						|
// Rewrite rewrites the current request based upon partial match of the
 | 
						|
// name in the question section of the request.
 | 
						|
func (rule *substringNameRule) Rewrite(ctx context.Context, state request.Request) Result {
 | 
						|
	if strings.Contains(state.Name(), rule.Substring) {
 | 
						|
		state.Req.Question[0].Name = strings.Replace(state.Name(), rule.Substring, rule.Replacement, -1)
 | 
						|
		return RewriteDone
 | 
						|
	}
 | 
						|
	return RewriteIgnored
 | 
						|
}
 | 
						|
 | 
						|
// Rewrite rewrites the current request when the name in the question
 | 
						|
// section of the request matches a regular expression.
 | 
						|
func (rule *regexNameRule) Rewrite(ctx context.Context, state request.Request) Result {
 | 
						|
	regexGroups := rule.Pattern.FindStringSubmatch(state.Name())
 | 
						|
	if len(regexGroups) == 0 {
 | 
						|
		return RewriteIgnored
 | 
						|
	}
 | 
						|
	s := rule.Replacement
 | 
						|
	for groupIndex, groupValue := range regexGroups {
 | 
						|
		groupIndexStr := "{" + strconv.Itoa(groupIndex) + "}"
 | 
						|
		if strings.Contains(s, groupIndexStr) {
 | 
						|
			s = strings.Replace(s, groupIndexStr, groupValue, -1)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	state.Req.Question[0].Name = s
 | 
						|
	return RewriteDone
 | 
						|
}
 | 
						|
 | 
						|
// newNameRule creates a name matching rule based on exact, partial, or regex match
 | 
						|
func newNameRule(nextAction string, args ...string) (Rule, error) {
 | 
						|
	var matchType, rewriteQuestionFrom, rewriteQuestionTo string
 | 
						|
	var rewriteAnswerField, rewriteAnswerFrom, rewriteAnswerTo string
 | 
						|
	if len(args) < 2 {
 | 
						|
		return nil, fmt.Errorf("too few arguments for a name rule")
 | 
						|
	}
 | 
						|
	if len(args) == 2 {
 | 
						|
		matchType = "exact"
 | 
						|
		rewriteQuestionFrom = plugin.Name(args[0]).Normalize()
 | 
						|
		rewriteQuestionTo = plugin.Name(args[1]).Normalize()
 | 
						|
	}
 | 
						|
	if len(args) >= 3 {
 | 
						|
		matchType = strings.ToLower(args[0])
 | 
						|
		rewriteQuestionFrom = plugin.Name(args[1]).Normalize()
 | 
						|
		rewriteQuestionTo = plugin.Name(args[2]).Normalize()
 | 
						|
	}
 | 
						|
	if matchType == RegexMatch {
 | 
						|
		rewriteQuestionFrom = args[1]
 | 
						|
		rewriteQuestionTo = args[2]
 | 
						|
	}
 | 
						|
	if matchType == ExactMatch || matchType == SuffixMatch {
 | 
						|
		if !hasClosingDot(rewriteQuestionFrom) {
 | 
						|
			rewriteQuestionFrom = rewriteQuestionFrom + "."
 | 
						|
		}
 | 
						|
		if !hasClosingDot(rewriteQuestionTo) {
 | 
						|
			rewriteQuestionTo = rewriteQuestionTo + "."
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if len(args) > 3 && len(args) != 7 {
 | 
						|
		return nil, fmt.Errorf("response rewrites must consist only of a name rule with 3 arguments and an answer rule with 3 arguments")
 | 
						|
	}
 | 
						|
 | 
						|
	if len(args) < 7 {
 | 
						|
		switch matchType {
 | 
						|
		case ExactMatch:
 | 
						|
			rewriteAnswerFromPattern, err := isValidRegexPattern(rewriteQuestionTo, rewriteQuestionFrom)
 | 
						|
			if err != nil {
 | 
						|
				return nil, err
 | 
						|
			}
 | 
						|
			return &exactNameRule{
 | 
						|
				nextAction,
 | 
						|
				rewriteQuestionFrom,
 | 
						|
				rewriteQuestionTo,
 | 
						|
				ResponseRule{
 | 
						|
					Active:      true,
 | 
						|
					Type:        "name",
 | 
						|
					Pattern:     rewriteAnswerFromPattern,
 | 
						|
					Replacement: rewriteQuestionFrom,
 | 
						|
				},
 | 
						|
			}, nil
 | 
						|
		case PrefixMatch:
 | 
						|
			return &prefixNameRule{
 | 
						|
				nextAction,
 | 
						|
				rewriteQuestionFrom,
 | 
						|
				rewriteQuestionTo,
 | 
						|
			}, nil
 | 
						|
		case SuffixMatch:
 | 
						|
			return &suffixNameRule{
 | 
						|
				nextAction,
 | 
						|
				rewriteQuestionFrom,
 | 
						|
				rewriteQuestionTo,
 | 
						|
			}, nil
 | 
						|
		case SubstringMatch:
 | 
						|
			return &substringNameRule{
 | 
						|
				nextAction,
 | 
						|
				rewriteQuestionFrom,
 | 
						|
				rewriteQuestionTo,
 | 
						|
			}, nil
 | 
						|
		case RegexMatch:
 | 
						|
			rewriteQuestionFromPattern, err := isValidRegexPattern(rewriteQuestionFrom, rewriteQuestionTo)
 | 
						|
			if err != nil {
 | 
						|
				return nil, err
 | 
						|
			}
 | 
						|
			rewriteQuestionTo := plugin.Name(args[2]).Normalize()
 | 
						|
			return ®exNameRule{
 | 
						|
				nextAction,
 | 
						|
				rewriteQuestionFromPattern,
 | 
						|
				rewriteQuestionTo,
 | 
						|
				ResponseRule{
 | 
						|
					Type: "name",
 | 
						|
				},
 | 
						|
			}, nil
 | 
						|
		default:
 | 
						|
			return nil, fmt.Errorf("A name rule supports only exact, prefix, suffix, substring, and regex name matching, received: %s", matchType)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if len(args) == 7 {
 | 
						|
		if matchType == RegexMatch {
 | 
						|
			if args[3] != "answer" {
 | 
						|
				return nil, fmt.Errorf("exceeded the number of arguments for a regex name rule")
 | 
						|
			}
 | 
						|
			rewriteQuestionFromPattern, err := isValidRegexPattern(rewriteQuestionFrom, rewriteQuestionTo)
 | 
						|
			if err != nil {
 | 
						|
				return nil, err
 | 
						|
			}
 | 
						|
			rewriteAnswerField = strings.ToLower(args[4])
 | 
						|
			switch rewriteAnswerField {
 | 
						|
			case "name":
 | 
						|
			default:
 | 
						|
				return nil, fmt.Errorf("exceeded the number of arguments for a regex name rule")
 | 
						|
			}
 | 
						|
			rewriteAnswerFrom = args[5]
 | 
						|
			rewriteAnswerTo = args[6]
 | 
						|
			rewriteAnswerFromPattern, err := isValidRegexPattern(rewriteAnswerFrom, rewriteAnswerTo)
 | 
						|
			if err != nil {
 | 
						|
				return nil, err
 | 
						|
			}
 | 
						|
			rewriteQuestionTo = plugin.Name(args[2]).Normalize()
 | 
						|
			rewriteAnswerTo = plugin.Name(args[6]).Normalize()
 | 
						|
			return ®exNameRule{
 | 
						|
				nextAction,
 | 
						|
				rewriteQuestionFromPattern,
 | 
						|
				rewriteQuestionTo,
 | 
						|
				ResponseRule{
 | 
						|
					Active:      true,
 | 
						|
					Type:        "name",
 | 
						|
					Pattern:     rewriteAnswerFromPattern,
 | 
						|
					Replacement: rewriteAnswerTo,
 | 
						|
				},
 | 
						|
			}, nil
 | 
						|
		}
 | 
						|
		return nil, fmt.Errorf("the rewrite of response is supported only for name regex rule")
 | 
						|
	}
 | 
						|
	return nil, fmt.Errorf("the rewrite rule is invalid: %s", args)
 | 
						|
}
 | 
						|
 | 
						|
// Mode returns the processing nextAction
 | 
						|
func (rule *exactNameRule) Mode() string     { return rule.NextAction }
 | 
						|
func (rule *prefixNameRule) Mode() string    { return rule.NextAction }
 | 
						|
func (rule *suffixNameRule) Mode() string    { return rule.NextAction }
 | 
						|
func (rule *substringNameRule) Mode() string { return rule.NextAction }
 | 
						|
func (rule *regexNameRule) Mode() string     { return rule.NextAction }
 | 
						|
 | 
						|
// GetResponseRule return a rule to rewrite the response with. Currently not implemented.
 | 
						|
func (rule *exactNameRule) GetResponseRule() ResponseRule { return rule.ResponseRule }
 | 
						|
 | 
						|
// GetResponseRule return a rule to rewrite the response with. Currently not implemented.
 | 
						|
func (rule *prefixNameRule) GetResponseRule() ResponseRule { return ResponseRule{} }
 | 
						|
 | 
						|
// GetResponseRule return a rule to rewrite the response with. Currently not implemented.
 | 
						|
func (rule *suffixNameRule) GetResponseRule() ResponseRule { return ResponseRule{} }
 | 
						|
 | 
						|
// GetResponseRule return a rule to rewrite the response with. Currently not implemented.
 | 
						|
func (rule *substringNameRule) GetResponseRule() ResponseRule { return ResponseRule{} }
 | 
						|
 | 
						|
// GetResponseRule return a rule to rewrite the response with.
 | 
						|
func (rule *regexNameRule) GetResponseRule() ResponseRule { return rule.ResponseRule }
 | 
						|
 | 
						|
// hasClosingDot return true if s has a closing dot at the end.
 | 
						|
func hasClosingDot(s string) bool {
 | 
						|
	if strings.HasSuffix(s, ".") {
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	return false
 | 
						|
}
 | 
						|
 | 
						|
// getSubExprUsage return the number of subexpressions used in s.
 | 
						|
func getSubExprUsage(s string) int {
 | 
						|
	subExprUsage := 0
 | 
						|
	for i := 0; i <= 100; i++ {
 | 
						|
		if strings.Contains(s, "{"+strconv.Itoa(i)+"}") {
 | 
						|
			subExprUsage++
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return subExprUsage
 | 
						|
}
 | 
						|
 | 
						|
// isValidRegexPattern return a regular expression for pattern matching or errors, if any.
 | 
						|
func isValidRegexPattern(rewriteFrom, rewriteTo string) (*regexp.Regexp, error) {
 | 
						|
	rewriteFromPattern, err := regexp.Compile(rewriteFrom)
 | 
						|
	if err != nil {
 | 
						|
		return nil, fmt.Errorf("invalid regex matching pattern: %s", rewriteFrom)
 | 
						|
	}
 | 
						|
	if getSubExprUsage(rewriteTo) > rewriteFromPattern.NumSubexp() {
 | 
						|
		return nil, fmt.Errorf("the rewrite regex pattern (%s) uses more subexpressions than its corresponding matching regex pattern (%s)", rewriteTo, rewriteFrom)
 | 
						|
	}
 | 
						|
	return rewriteFromPattern, nil
 | 
						|
}
 |