mirror of
				https://github.com/coredns/coredns.git
				synced 2025-11-03 10:43:20 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			225 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			225 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package rewrite
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"fmt"
 | 
						|
	"regexp"
 | 
						|
	"strconv"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	"github.com/coredns/coredns/plugin"
 | 
						|
	"github.com/coredns/coredns/request"
 | 
						|
	//"github.com/miekg/dns"
 | 
						|
)
 | 
						|
 | 
						|
type exactTtlRule struct {
 | 
						|
	NextAction string
 | 
						|
	From       string
 | 
						|
	ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
type prefixTtlRule struct {
 | 
						|
	NextAction string
 | 
						|
	Prefix     string
 | 
						|
	ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
type suffixTtlRule struct {
 | 
						|
	NextAction string
 | 
						|
	Suffix     string
 | 
						|
	ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
type substringTtlRule struct {
 | 
						|
	NextAction string
 | 
						|
	Substring  string
 | 
						|
	ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
type regexTtlRule struct {
 | 
						|
	NextAction string
 | 
						|
	Pattern    *regexp.Regexp
 | 
						|
	ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
// Rewrite rewrites the current request based upon exact match of the name
 | 
						|
// in the question section of the request.
 | 
						|
func (rule *exactTtlRule) Rewrite(ctx context.Context, state request.Request) Result {
 | 
						|
	if rule.From == state.Name() {
 | 
						|
		return RewriteDone
 | 
						|
	}
 | 
						|
	return RewriteIgnored
 | 
						|
}
 | 
						|
 | 
						|
// Rewrite rewrites the current request when the name begins with the matching string.
 | 
						|
func (rule *prefixTtlRule) Rewrite(ctx context.Context, state request.Request) Result {
 | 
						|
	if strings.HasPrefix(state.Name(), rule.Prefix) {
 | 
						|
		return RewriteDone
 | 
						|
	}
 | 
						|
	return RewriteIgnored
 | 
						|
}
 | 
						|
 | 
						|
// Rewrite rewrites the current request when the name ends with the matching string.
 | 
						|
func (rule *suffixTtlRule) Rewrite(ctx context.Context, state request.Request) Result {
 | 
						|
	if strings.HasSuffix(state.Name(), rule.Suffix) {
 | 
						|
		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 *substringTtlRule) Rewrite(ctx context.Context, state request.Request) Result {
 | 
						|
	if strings.Contains(state.Name(), rule.Substring) {
 | 
						|
		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 *regexTtlRule) Rewrite(ctx context.Context, state request.Request) Result {
 | 
						|
	regexGroups := rule.Pattern.FindStringSubmatch(state.Name())
 | 
						|
	if len(regexGroups) == 0 {
 | 
						|
		return RewriteIgnored
 | 
						|
	}
 | 
						|
	return RewriteDone
 | 
						|
}
 | 
						|
 | 
						|
// newTtlRule creates a name matching rule based on exact, partial, or regex match
 | 
						|
func newTtlRule(nextAction string, args ...string) (Rule, error) {
 | 
						|
	if len(args) < 2 {
 | 
						|
		return nil, fmt.Errorf("too few (%d) arguments for a ttl rule", len(args))
 | 
						|
	}
 | 
						|
	var s string
 | 
						|
	if len(args) == 2 {
 | 
						|
		s = args[1]
 | 
						|
	}
 | 
						|
	if len(args) == 3 {
 | 
						|
		s = args[2]
 | 
						|
	}
 | 
						|
	ttl, valid := isValidTtl(s)
 | 
						|
	if valid == false {
 | 
						|
		return nil, fmt.Errorf("invalid TTL '%s' for a ttl rule", s)
 | 
						|
	}
 | 
						|
	if len(args) == 3 {
 | 
						|
		switch strings.ToLower(args[0]) {
 | 
						|
		case ExactMatch:
 | 
						|
			return &exactTtlRule{
 | 
						|
				nextAction,
 | 
						|
				plugin.Name(args[1]).Normalize(),
 | 
						|
				ResponseRule{
 | 
						|
					Active: true,
 | 
						|
					Type:   "ttl",
 | 
						|
					Ttl:    ttl,
 | 
						|
				},
 | 
						|
			}, nil
 | 
						|
		case PrefixMatch:
 | 
						|
			return &prefixTtlRule{
 | 
						|
				nextAction,
 | 
						|
				plugin.Name(args[1]).Normalize(),
 | 
						|
				ResponseRule{
 | 
						|
					Active: true,
 | 
						|
					Type:   "ttl",
 | 
						|
					Ttl:    ttl,
 | 
						|
				},
 | 
						|
			}, nil
 | 
						|
		case SuffixMatch:
 | 
						|
			return &suffixTtlRule{
 | 
						|
				nextAction,
 | 
						|
				plugin.Name(args[1]).Normalize(),
 | 
						|
				ResponseRule{
 | 
						|
					Active: true,
 | 
						|
					Type:   "ttl",
 | 
						|
					Ttl:    ttl,
 | 
						|
				},
 | 
						|
			}, nil
 | 
						|
		case SubstringMatch:
 | 
						|
			return &substringTtlRule{
 | 
						|
				nextAction,
 | 
						|
				plugin.Name(args[1]).Normalize(),
 | 
						|
				ResponseRule{
 | 
						|
					Active: true,
 | 
						|
					Type:   "ttl",
 | 
						|
					Ttl:    ttl,
 | 
						|
				},
 | 
						|
			}, nil
 | 
						|
		case RegexMatch:
 | 
						|
			regexPattern, err := regexp.Compile(args[1])
 | 
						|
			if err != nil {
 | 
						|
				return nil, fmt.Errorf("Invalid regex pattern in a ttl rule: %s", args[1])
 | 
						|
			}
 | 
						|
			return ®exTtlRule{
 | 
						|
				nextAction,
 | 
						|
				regexPattern,
 | 
						|
				ResponseRule{
 | 
						|
					Active: true,
 | 
						|
					Type:   "ttl",
 | 
						|
					Ttl:    ttl,
 | 
						|
				},
 | 
						|
			}, nil
 | 
						|
		default:
 | 
						|
			return nil, fmt.Errorf("A ttl rule supports only exact, prefix, suffix, substring, and regex name matching")
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if len(args) > 3 {
 | 
						|
		return nil, fmt.Errorf("many few arguments for a ttl rule")
 | 
						|
	}
 | 
						|
	return &exactTtlRule{
 | 
						|
		nextAction,
 | 
						|
		plugin.Name(args[0]).Normalize(),
 | 
						|
		ResponseRule{
 | 
						|
			Active: true,
 | 
						|
			Type:   "ttl",
 | 
						|
			Ttl:    ttl,
 | 
						|
		},
 | 
						|
	}, nil
 | 
						|
}
 | 
						|
 | 
						|
// Mode returns the processing nextAction
 | 
						|
func (rule *exactTtlRule) Mode() string     { return rule.NextAction }
 | 
						|
func (rule *prefixTtlRule) Mode() string    { return rule.NextAction }
 | 
						|
func (rule *suffixTtlRule) Mode() string    { return rule.NextAction }
 | 
						|
func (rule *substringTtlRule) Mode() string { return rule.NextAction }
 | 
						|
func (rule *regexTtlRule) Mode() string     { return rule.NextAction }
 | 
						|
 | 
						|
// GetResponseRule return a rule to rewrite the response with. Currently not implemented.
 | 
						|
func (rule *exactTtlRule) GetResponseRule() ResponseRule {
 | 
						|
	return rule.ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
// GetResponseRule return a rule to rewrite the response with. Currently not implemented.
 | 
						|
func (rule *prefixTtlRule) GetResponseRule() ResponseRule {
 | 
						|
	return rule.ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
// GetResponseRule return a rule to rewrite the response with. Currently not implemented.
 | 
						|
func (rule *suffixTtlRule) GetResponseRule() ResponseRule {
 | 
						|
	return rule.ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
// GetResponseRule return a rule to rewrite the response with. Currently not implemented.
 | 
						|
func (rule *substringTtlRule) GetResponseRule() ResponseRule {
 | 
						|
	return rule.ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
// GetResponseRule return a rule to rewrite the response with.
 | 
						|
func (rule *regexTtlRule) GetResponseRule() ResponseRule {
 | 
						|
	return rule.ResponseRule
 | 
						|
}
 | 
						|
 | 
						|
// validTtl returns true if v is valid TTL value.
 | 
						|
func isValidTtl(v string) (uint32, bool) {
 | 
						|
	i, err := strconv.Atoi(v)
 | 
						|
	if err != nil {
 | 
						|
		return uint32(0), false
 | 
						|
	}
 | 
						|
	if i > 2147483647 {
 | 
						|
		return uint32(0), false
 | 
						|
	}
 | 
						|
	if i < 0 {
 | 
						|
		return uint32(0), false
 | 
						|
	}
 | 
						|
	return uint32(i), true
 | 
						|
}
 |