mirror of
https://github.com/coredns/coredns.git
synced 2025-10-23 21:31:59 +02:00
From [Golang coding convention](https://github.com/golang/go/wiki/CodeReviewComments#error-strings) Co-Authored-By: Nguyen Van Trung [trungnvfet@outlook.com](mailto:trungnvfet@outlook.com) Signed-off-by: Nguyen Quang Huy [huynq0911@gmail.com](mailto:huynq0911@gmail.com)
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
|
|
}
|